Skip to content

Commit

Permalink
[microNPU] enable USMP (apache#10022)
Browse files Browse the repository at this point in the history
This commit enables USMP in the microNPU codegen
and tests. The microNPU codegen is modified to
support Let nodes that are produced as from USMP.
  • Loading branch information
manupak authored and pfk-beta committed Apr 11, 2022
1 parent 2ce4eb4 commit e56b0f8
Show file tree
Hide file tree
Showing 12 changed files with 333 additions and 149 deletions.
2 changes: 1 addition & 1 deletion apps/microtvm/zephyr_cmsisnn/src/main.c
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ extern float output_storage[12];

extern const size_t output_len;

static uint8_t g_crt_workspace[TVMGEN_DEFAULT_WORKSPACE_SIZE + 256];
static uint8_t g_crt_workspace[TVMGEN_DEFAULT_WORKSPACE_SIZE + 512];
tvm_workspace_t app_workspace;

void TVMLogf(const char* msg, ...) {
Expand Down
13 changes: 7 additions & 6 deletions include/tvm/tir/usmp/utils.h
Original file line number Diff line number Diff line change
Expand Up @@ -185,24 +185,24 @@ struct AllocatedPoolInfoNode : public Object {
PoolInfo pool_info;
/*! \brief The allocated size into this pool */
Integer allocated_size;
/*! \brief An optional associated pool Var*/
Optional<Var> pool_var;
/*! \brief An optional associated pool Var index of PrimFunc params*/
Optional<Integer> pool_var_idx;

void VisitAttrs(tvm::AttrVisitor* v) {
v->Visit("pool_info", &pool_info);
v->Visit("allocated_size", &allocated_size);
v->Visit("pool_var", &pool_var);
v->Visit("pool_var_idx", &pool_var_idx);
}

bool SEqualReduce(const AllocatedPoolInfoNode* other, SEqualReducer equal) const {
return equal(pool_info, other->pool_info) && equal(allocated_size, other->allocated_size) &&
equal(pool_var, other->pool_var);
equal(pool_var_idx, other->pool_var_idx);
}

void SHashReduce(SHashReducer hash_reduce) const {
hash_reduce(pool_info);
hash_reduce(allocated_size);
hash_reduce(pool_var);
hash_reduce(pool_var_idx);
}

static constexpr const char* _type_key = "tir.usmp.AllocatedPoolInfo";
Expand All @@ -211,7 +211,8 @@ struct AllocatedPoolInfoNode : public Object {

class AllocatedPoolInfo : public ObjectRef {
public:
TVM_DLL AllocatedPoolInfo(PoolInfo pool_info, Integer allocated_size, Var pool_var = Var());
TVM_DLL AllocatedPoolInfo(PoolInfo pool_info, Integer allocated_size,
Integer pool_var_idx = Integer());
TVM_DEFINE_MUTABLE_OBJECT_REF_METHODS(AllocatedPoolInfo, ObjectRef, AllocatedPoolInfoNode);
};

Expand Down
91 changes: 48 additions & 43 deletions python/tvm/micro/model_library_format.py
Original file line number Diff line number Diff line change
Expand Up @@ -181,42 +181,26 @@ def _build_function_memory_map(function_metadata):
"""
device_max_workspace = dict()
main_func_metadata = function_metadata[MAIN_FUNC_NAME_STR]
num_targets = len(main_func_metadata.workspace_sizes.items())
from tvm.driver import tvmc # pylint: disable=import-outside-toplevel

external_codegens = tvmc.composite_target.get_codegen_names()
func_entries = []
target_local_entries = dict()
for i in range(num_targets):
main_target = main_func_metadata.workspace_sizes.items()[i][0]
device_max_workspace[main_target] = 0
for func_name, finfo in function_metadata.items():
if func_name == MAIN_FUNC_NAME_STR:
continue
target_local_entries[func_name] = list()

for func_name, finfo in function_metadata.items():
# Skip a few unsupported cases:
# 1. The main function metadata is exported elsewhere.
# 2. BYOC operator implementations do not currently export useful FunctionInfo.
if func_name == MAIN_FUNC_NAME_STR or not finfo.tir_primfuncs:
continue
assert (
len(finfo.constant_sizes.items()) == num_targets
), f"{func_name}: found {finfo.constant_sizes!r} vs {num_targets}"
assert len(finfo.io_sizes.items()) == num_targets
target = finfo.workspace_sizes.items()[i][0]
workspace_size = finfo.workspace_sizes.items()[i][1]
for func_name, finfo in function_metadata.items():
# Skip a few unsupported cases:
# 1. The main function metadata is exported elsewhere.
# 2. BYOC operator implementations do not currently export useful FunctionInfo.
if func_name == MAIN_FUNC_NAME_STR or not finfo.tir_primfuncs:
continue
if func_name not in target_local_entries.keys():
target_local_entries[func_name] = list()
for target in dict(finfo.workspace_sizes).keys():
workspace_size = finfo.workspace_sizes[target]
target_entry = {
"device": int(target.kind.device_type),
"workspace_size_bytes": int(workspace_size),
}
target_local_entries[func_name].append(target_entry)
if workspace_size > device_max_workspace.get(target, 0):
device_max_workspace[target] = workspace_size
# TODO(Mousius) - Remove this massive hack when Targets are unified
if target.kind.name in external_codegens:
device_max_workspace[main_target] += int(workspace_size)
if workspace_size >= device_max_workspace.get(int(target.kind.device_type), 0):
device_max_workspace[int(target.kind.device_type)] = workspace_size

for func_name, target_entries_ in target_local_entries.items():
func_entry = {
Expand All @@ -225,25 +209,46 @@ def _build_function_memory_map(function_metadata):
}
func_entries.append(func_entry)

target_main_entries = list()
for i in range(num_targets):
target = main_func_metadata.workspace_sizes.items()[i][0]
main_func_local_workspace = main_func_metadata.workspace_sizes.items()[i][1]
main_func_constants = main_func_metadata.constant_sizes.items()[i][1]
main_func_io = main_func_metadata.io_sizes.items()[i][1]
target_main_entries.append(
{
"device": int(target.kind.device_type),
"workspace_size_bytes": int(device_max_workspace[target])
+ int(main_func_local_workspace),
"constants_size_bytes": int(main_func_constants),
"io_size_bytes": int(main_func_io),
}
target_main_entries = dict()

def _create_empty_entry(target_device_type):
return {
"device": int(target_device_type),
"workspace_size_bytes": 0,
"constants_size_bytes": 0,
"io_size_bytes": 0,
}

for target in dict(main_func_metadata.workspace_sizes).keys():
main_func_local_workspace = main_func_metadata.workspace_sizes[target]
target_main_entries[int(target.kind.device_type)] = _create_empty_entry(
int(target.kind.device_type)
)
target_main_entries[int(target.kind.device_type)]["workspace_size_bytes"] = int(
device_max_workspace.get(int(target.kind.device_type), 0)
) + int(main_func_local_workspace)

for target in dict(main_func_metadata.constant_sizes).keys():
if int(target.kind.device_type) not in target_main_entries.keys():
target_main_entries[int(target.kind.device_type)] = _create_empty_entry(
int(target.kind.device_type)
)
target_main_entries[int(target.kind.device_type)]["constants_size_bytes"] = int(
main_func_metadata.constant_sizes[target]
)

for target in dict(main_func_metadata.io_sizes).keys():
if int(target.kind.device_type) not in target_main_entries.keys():
target_main_entries[int(target.kind.device_type)] = _create_empty_entry(
int(target.kind.device_type)
)
target_main_entries[int(target.kind.device_type)]["io_size_bytes"] = int(
main_func_metadata.io_sizes[target]
)

ret = {
"operator_functions": func_entries,
"main": target_main_entries,
"main": list(target_main_entries.values()),
}
return ret

Expand Down
Loading

0 comments on commit e56b0f8

Please sign in to comment.