Allow not copying the wasm binary in wasm-c-api and not referring to the binary in wasm/aot loader (#3389)
Add flag `LoadArgs.clone_wasm_binary` to control whether to clone the wasm/aot binary in wasm-c-api module. If false, API `wasm_module_new_ex` won't clone the binary, which may reduce the footprint. Add flag `LoadArgs.wasm_binary_freeable` to control whether the wasm/aot binary may be freed after instantiation for wamr API `wasm_runtime_load_ex`, if yes, then for some running modes, the wasm/aot module doesn't refer to the input binary again so developer can free it after instantiation to reduce the footprint. And add API `wasm_module_is_underlying_binary_freeable` and `wasm_runtime_is_underlying_binary_freeable` to check whether the input binary can be freed after instantiation for wasm-c-api and wamr api. And add sample to illustrate it.
This commit is contained in:
@ -42,6 +42,8 @@
|
||||
typedef struct wasm_module_ex_t {
|
||||
struct WASMModuleCommon *module_comm_rt;
|
||||
wasm_byte_vec_t *binary;
|
||||
/* If true, binary in wasm_module_ex_t contains a copy of the WASM binary */
|
||||
bool is_binary_cloned;
|
||||
korp_mutex lock;
|
||||
uint32 ref_count;
|
||||
#if WASM_ENABLE_WASM_CACHE != 0
|
||||
@ -2242,8 +2244,7 @@ quit:
|
||||
#endif /* WASM_ENABLE_WASM_CACHE != 0 */
|
||||
|
||||
wasm_module_t *
|
||||
wasm_module_new_ex(wasm_store_t *store, const wasm_byte_vec_t *binary,
|
||||
const LoadArgs *args)
|
||||
wasm_module_new_ex(wasm_store_t *store, wasm_byte_vec_t *binary, LoadArgs *args)
|
||||
{
|
||||
char error_buf[128] = { 0 };
|
||||
wasm_module_ex_t *module_ex = NULL;
|
||||
@ -2291,14 +2292,21 @@ wasm_module_new_ex(wasm_store_t *store, const wasm_byte_vec_t *binary,
|
||||
if (!module_ex)
|
||||
goto quit;
|
||||
|
||||
module_ex->binary = malloc_internal(sizeof(wasm_byte_vec_t));
|
||||
if (!module_ex->binary)
|
||||
goto free_module;
|
||||
module_ex->is_binary_cloned = args->clone_wasm_binary;
|
||||
if (args->clone_wasm_binary) {
|
||||
module_ex->binary = malloc_internal(sizeof(wasm_byte_vec_t));
|
||||
if (!module_ex->binary)
|
||||
goto free_module;
|
||||
|
||||
wasm_byte_vec_copy(module_ex->binary, binary);
|
||||
if (!module_ex->binary->data)
|
||||
goto free_binary;
|
||||
wasm_byte_vec_copy(module_ex->binary, binary);
|
||||
if (!module_ex->binary->data)
|
||||
goto free_binary;
|
||||
}
|
||||
else {
|
||||
module_ex->binary = binary;
|
||||
}
|
||||
|
||||
args->wasm_binary_freeable = !args->clone_wasm_binary;
|
||||
module_ex->module_comm_rt = wasm_runtime_load_ex(
|
||||
(uint8 *)module_ex->binary->data, (uint32)module_ex->binary->size, args,
|
||||
error_buf, (uint32)sizeof(error_buf));
|
||||
@ -2336,9 +2344,11 @@ remove_last:
|
||||
unload:
|
||||
wasm_runtime_unload(module_ex->module_comm_rt);
|
||||
free_vec:
|
||||
wasm_byte_vec_delete(module_ex->binary);
|
||||
if (args->clone_wasm_binary)
|
||||
wasm_byte_vec_delete(module_ex->binary);
|
||||
free_binary:
|
||||
wasm_runtime_free(module_ex->binary);
|
||||
if (args->clone_wasm_binary)
|
||||
wasm_runtime_free(module_ex->binary);
|
||||
free_module:
|
||||
wasm_runtime_free(module_ex);
|
||||
quit:
|
||||
@ -2351,7 +2361,8 @@ wasm_module_new(wasm_store_t *store, const wasm_byte_vec_t *binary)
|
||||
{
|
||||
LoadArgs args = { 0 };
|
||||
args.name = "";
|
||||
return wasm_module_new_ex(store, binary, &args);
|
||||
args.clone_wasm_binary = true;
|
||||
return wasm_module_new_ex(store, (wasm_byte_vec_t *)binary, &args);
|
||||
}
|
||||
|
||||
bool
|
||||
@ -2410,7 +2421,8 @@ wasm_module_delete_internal(wasm_module_t *module)
|
||||
return;
|
||||
}
|
||||
|
||||
DEINIT_VEC(module_ex->binary, wasm_byte_vec_delete);
|
||||
if (module_ex->is_binary_cloned)
|
||||
DEINIT_VEC(module_ex->binary, wasm_byte_vec_delete);
|
||||
|
||||
if (module_ex->module_comm_rt) {
|
||||
wasm_runtime_unload(module_ex->module_comm_rt);
|
||||
@ -2994,6 +3006,16 @@ wasm_module_get_name(wasm_module_t *module)
|
||||
return wasm_runtime_get_module_name(module_ex->module_comm_rt);
|
||||
}
|
||||
|
||||
bool
|
||||
wasm_module_is_underlying_binary_freeable(
|
||||
const wasm_module_t *module, const struct wasm_instance_t *instance)
|
||||
{
|
||||
if (((wasm_module_ex_t *)module)->is_binary_cloned)
|
||||
return true;
|
||||
|
||||
return wasm_runtime_is_underlying_binary_freeable(instance->inst_comm_rt);
|
||||
}
|
||||
|
||||
static wasm_func_t *
|
||||
wasm_func_new_basic(wasm_store_t *store, const wasm_functype_t *type,
|
||||
wasm_func_callback_t func_callback)
|
||||
|
||||
@ -1350,12 +1350,18 @@ wasm_runtime_load_ex(uint8 *buf, uint32 size, const LoadArgs *args,
|
||||
true,
|
||||
#endif
|
||||
args, error_buf, error_buf_size);
|
||||
if (module_common)
|
||||
((WASMModule *)module_common)->is_binary_freeable =
|
||||
args->wasm_binary_freeable;
|
||||
#endif
|
||||
}
|
||||
else if (get_package_type(buf, size) == Wasm_Module_AoT) {
|
||||
#if WASM_ENABLE_AOT != 0
|
||||
module_common = (WASMModuleCommon *)aot_load_from_aot_file(
|
||||
buf, size, args, error_buf, error_buf_size);
|
||||
if (module_common)
|
||||
((AOTModule *)module_common)->is_binary_freeable =
|
||||
args->wasm_binary_freeable;
|
||||
#endif
|
||||
}
|
||||
else {
|
||||
@ -1383,6 +1389,7 @@ wasm_runtime_load(uint8 *buf, uint32 size, char *error_buf,
|
||||
{
|
||||
LoadArgs args = { 0 };
|
||||
args.name = "";
|
||||
args.wasm_binary_freeable = false;
|
||||
return wasm_runtime_load_ex(buf, size, &args, error_buf, error_buf_size);
|
||||
}
|
||||
|
||||
@ -1400,6 +1407,7 @@ wasm_runtime_load_from_sections(WASMSection *section_list, bool is_aot,
|
||||
LOG_DEBUG("WASM module load failed from sections");
|
||||
return NULL;
|
||||
}
|
||||
((WASMModule *)module_common)->is_binary_freeable = true;
|
||||
return register_module_with_null_name(module_common, error_buf,
|
||||
error_buf_size);
|
||||
#endif
|
||||
@ -1412,6 +1420,7 @@ wasm_runtime_load_from_sections(WASMSection *section_list, bool is_aot,
|
||||
LOG_DEBUG("WASM module load failed from sections");
|
||||
return NULL;
|
||||
}
|
||||
((AOTModule *)module_common)->is_binary_freeable = true;
|
||||
return register_module_with_null_name(module_common, error_buf,
|
||||
error_buf_size);
|
||||
#endif
|
||||
@ -7247,3 +7256,51 @@ wasm_runtime_detect_native_stack_overflow_size(WASMExecEnv *exec_env,
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
WASM_RUNTIME_API_EXTERN bool
|
||||
wasm_runtime_is_underlying_binary_freeable(const wasm_module_inst_t module_inst)
|
||||
{
|
||||
uint32 i;
|
||||
|
||||
#if WASM_ENABLE_INTERP != 0
|
||||
if (module_inst->module_type == Wasm_Module_Bytecode) {
|
||||
#if (WASM_ENABLE_JIT != 0 || WASM_ENABLE_FAST_JIT != 0) \
|
||||
&& (WASM_ENABLE_LAZY_JIT != 0)
|
||||
return false;
|
||||
#elif WASM_ENABLE_FAST_INTERP == 0
|
||||
return false;
|
||||
#else
|
||||
/* Fast interpreter mode */
|
||||
WASMModule *module = ((WASMModuleInstance *)module_inst)->module;
|
||||
if (!module->is_binary_freeable)
|
||||
return false;
|
||||
#if WASM_ENABLE_GC != 0 && WASM_ENABLE_STRINGREF != 0
|
||||
if (module->string_literal_ptrs)
|
||||
return false;
|
||||
#endif
|
||||
#if WASM_ENABLE_BULK_MEMORY != 0
|
||||
for (i = 0; i < module->data_seg_count; i++)
|
||||
if (!bh_bitmap_get_bit(
|
||||
((WASMModuleInstance *)module_inst)->e->common.data_dropped,
|
||||
i))
|
||||
return false;
|
||||
#endif
|
||||
#endif
|
||||
}
|
||||
#endif /* WASM_ENABLE_INTERP != 0 */
|
||||
#if WASM_ENABLE_AOT != 0
|
||||
if (module_inst->module_type == Wasm_Module_AoT) {
|
||||
AOTModule *module =
|
||||
(AOTModule *)((AOTModuleInstance *)module_inst)->module;
|
||||
if (!module->is_binary_freeable)
|
||||
return false;
|
||||
#if WASM_ENABLE_GC != 0 && WASM_ENABLE_STRINGREF != 0
|
||||
if (module->string_literal_ptrs)
|
||||
return false;
|
||||
#endif
|
||||
}
|
||||
#endif /* WASM_ENABLE_AOT != 0 */
|
||||
|
||||
(void)i;
|
||||
return true;
|
||||
}
|
||||
|
||||
@ -1199,6 +1199,10 @@ WASM_RUNTIME_API_EXTERN bool
|
||||
wasm_runtime_detect_native_stack_overflow_size(WASMExecEnv *exec_env,
|
||||
uint32 requested_size);
|
||||
|
||||
WASM_RUNTIME_API_EXTERN bool
|
||||
wasm_runtime_is_underlying_binary_freeable(
|
||||
const wasm_module_inst_t module_inst);
|
||||
|
||||
#if WASM_ENABLE_LINUX_PERF != 0
|
||||
bool
|
||||
wasm_runtime_get_linux_perf(void);
|
||||
|
||||
Reference in New Issue
Block a user