wasm_native.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603
  1. /*
  2. * Copyright (C) 2019 Intel Corporation. All rights reserved.
  3. * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. */
  5. #include "wasm_native.h"
  6. #include "wasm_runtime_common.h"
  7. #include "bh_log.h"
  8. #if !defined(BH_PLATFORM_ZEPHYR) && !defined(BH_PLATFORM_ALIOS_THINGS) \
  9. && !defined(BH_PLATFORM_OPENRTOS) && !defined(BH_PLATFORM_ESP_IDF)
  10. #define ENABLE_QUICKSORT 1
  11. #else
  12. #define ENABLE_QUICKSORT 0
  13. #endif
  14. #define ENABLE_SORT_DEBUG 0
  15. #if ENABLE_SORT_DEBUG != 0
  16. #include <sys/time.h>
  17. #endif
  18. static NativeSymbolsVec g_native_symbols_vec = NULL;
  19. static uint32 g_native_libs_count = 0;
  20. static uint32 g_native_libs_size = 0;
  21. enum {
  22. #if WASM_ENABLE_LIBC_BUILTIN != 0
  23. ID_LIBC_BUILTIN = 0,
  24. #endif
  25. #if WASM_ENABLE_SPEC_TEST
  26. ID_SPECTEST,
  27. #endif
  28. #if WASM_ENABLE_LIBC_WASI != 0
  29. ID_LIBC_WASI_UNSTABLE,
  30. ID_LIBC_WASI_PREVIEW1,
  31. #endif
  32. #if WASM_ENABLE_BASE_LIB != 0
  33. ID_BASE_LIB,
  34. #endif
  35. #if WASM_ENABLE_APP_FRAMEWORK != 0
  36. ID_APP_FRAME,
  37. #endif
  38. #if WASM_ENABLE_LIB_PTHREAD != 0
  39. ID_PTHREAD,
  40. #endif
  41. #if WASM_ENABLE_LIBC_EMCC != 0
  42. ID_LIBC_EMCC,
  43. #endif
  44. ID_USER,
  45. NODE_COUNT
  46. };
  47. uint32
  48. get_libc_builtin_export_apis(NativeSymbol **p_libc_builtin_apis);
  49. #if WASM_ENABLE_SPEC_TEST != 0
  50. uint32
  51. get_spectest_export_apis(NativeSymbol **p_libc_builtin_apis);
  52. #endif
  53. uint32
  54. get_libc_wasi_export_apis(NativeSymbol **p_libc_wasi_apis);
  55. uint32
  56. get_base_lib_export_apis(NativeSymbol **p_base_lib_apis);
  57. uint32
  58. get_ext_lib_export_apis(NativeSymbol **p_ext_lib_apis);
  59. #if WASM_ENABLE_LIB_PTHREAD != 0
  60. bool
  61. lib_pthread_init();
  62. void
  63. lib_pthread_destroy();
  64. uint32
  65. get_lib_pthread_export_apis(NativeSymbol **p_lib_pthread_apis);
  66. #endif
  67. uint32
  68. get_libc_emcc_export_apis(NativeSymbol **p_libc_emcc_apis);
  69. bool
  70. check_symbol_signature(const WASMType *type, const char *signature)
  71. {
  72. const char *p = signature, *p_end;
  73. char sig_map[] = { 'F', 'f', 'I', 'i' }, sig;
  74. uint32 i = 0;
  75. if (!p || strlen(p) < 2)
  76. return false;
  77. p_end = p + strlen(signature);
  78. if (*p++ != '(')
  79. return false;
  80. if ((uint32)(p_end - p) < (uint32)(type->param_count + 1))
  81. /* signatures of parameters, and ')' */
  82. return false;
  83. for (i = 0; i < type->param_count; i++) {
  84. sig = *p++;
  85. if ((type->types[i] >= VALUE_TYPE_F64
  86. && type->types[i] <= VALUE_TYPE_I32
  87. && sig == sig_map[type->types[i] - VALUE_TYPE_F64])
  88. #if WASM_ENABLE_REF_TYPES != 0
  89. || (sig == 'i' && type->types[i] == VALUE_TYPE_EXTERNREF)
  90. #endif
  91. )
  92. /* normal parameter */
  93. continue;
  94. if (type->types[i] != VALUE_TYPE_I32)
  95. /* pointer and string must be i32 type */
  96. return false;
  97. if (sig == '*') {
  98. /* it is a pointer */
  99. if (i + 1 < type->param_count
  100. && type->types[i + 1] == VALUE_TYPE_I32 && *p == '~') {
  101. /* pointer length followed */
  102. i++;
  103. p++;
  104. }
  105. }
  106. else if (sig == '$') {
  107. /* it is a string */
  108. }
  109. else {
  110. /* invalid signature */
  111. return false;
  112. }
  113. }
  114. if (*p++ != ')')
  115. return false;
  116. if (type->result_count) {
  117. if (p >= p_end)
  118. return false;
  119. if (*p++ != sig_map[type->types[i] - VALUE_TYPE_F64])
  120. return false;
  121. }
  122. if (*p != '\0')
  123. return false;
  124. return true;
  125. }
  126. /*
  127. #if ENABLE_QUICKSORT == 0
  128. static void
  129. sort_symbol_ptr(NativeSymbol *native_symbols, uint32 n_native_symbols)
  130. {
  131. uint32 i, j;
  132. NativeSymbol temp;
  133. for (i = 0; i < n_native_symbols - 1; i++) {
  134. for (j = i + 1; j < n_native_symbols; j++) {
  135. if (strcmp(native_symbols[i].symbol, native_symbols[j].symbol)
  136. > 0) {
  137. temp = native_symbols[i];
  138. native_symbols[i] = native_symbols[j];
  139. native_symbols[j] = temp;
  140. }
  141. }
  142. }
  143. }
  144. #else
  145. static void
  146. swap_symbol(NativeSymbol *left, NativeSymbol *right)
  147. {
  148. NativeSymbol temp = *left;
  149. *left = *right;
  150. *right = temp;
  151. }
  152. static void
  153. quick_sort_symbols(NativeSymbol *native_symbols, int left, int right)
  154. {
  155. NativeSymbol base_symbol;
  156. int pin_left = left;
  157. int pin_right = right;
  158. if (left >= right) {
  159. return;
  160. }
  161. base_symbol = native_symbols[left];
  162. while (left < right) {
  163. while (left < right
  164. && strcmp(native_symbols[right].symbol, base_symbol.symbol)
  165. > 0) {
  166. right--;
  167. }
  168. if (left < right) {
  169. swap_symbol(&native_symbols[left], &native_symbols[right]);
  170. left++;
  171. }
  172. while (left < right
  173. && strcmp(native_symbols[left].symbol, base_symbol.symbol) < 0) {
  174. left++;
  175. }
  176. if (left < right) {
  177. swap_symbol(&native_symbols[left], &native_symbols[right]);
  178. right--;
  179. }
  180. }
  181. native_symbols[left] = base_symbol;
  182. quick_sort_symbols(native_symbols, pin_left, left - 1);
  183. quick_sort_symbols(native_symbols, left + 1, pin_right);
  184. }
  185. #endif
  186. */
  187. static void *
  188. lookup_symbol(NativeSymbol *native_symbols, uint32 n_native_symbols,
  189. const ConstStrDescription *symbol, const char **p_signature, void **p_attachment)
  190. {/*
  191. int low = 0, mid, ret;
  192. int high = (int32)n_native_symbols - 1;
  193. while (low <= high) {
  194. mid = (low + high) / 2;
  195. ret = strcmp(symbol, native_symbols[mid].symbol);
  196. if (ret == 0) {
  197. *p_signature = native_symbols[mid].signature;
  198. *p_attachment = native_symbols[mid].attachment;
  199. return native_symbols[mid].func_ptr;
  200. }
  201. else if (ret < 0)
  202. high = mid - 1;
  203. else
  204. low = mid + 1;
  205. }
  206. */
  207. uint32 i = 0;
  208. for (i = 0; i < n_native_symbols; i ++) {
  209. NativeSymbol * native_symbol = &native_symbols[i];
  210. if (native_symbol->u.symbol == symbol) {
  211. *p_signature = native_symbol->signature;
  212. *p_attachment = native_symbol->attachment;
  213. return native_symbol->func_ptr;
  214. }
  215. }
  216. return NULL;
  217. }
  218. void *
  219. wasm_native_resolve_symbol(const ConstStrDescription *module_name,
  220. const ConstStrDescription ** p_field_name,
  221. const WASMType *func_type, const char **p_signature,
  222. void **p_attachment, bool *p_call_conv_raw)
  223. {
  224. WASMRuntime * runtime = wasm_runtime_get_runtime();
  225. NativeSymbolsNode *node = NULL;
  226. const char *signature = NULL;
  227. const ConstStrDescription * field_name = *p_field_name;
  228. void *func_ptr = NULL, *attachment;
  229. uint32 field_id =0, first_sym_id = 0, symbol_index = 0;
  230. bool pass_check = false;
  231. if (wasm_runtime_is_system_symbol(runtime, module_name) &&
  232. wasm_runtime_is_system_symbol(runtime, field_name)) {
  233. for (uint32 i = 0; i < ID_USER; i ++) {
  234. node = &g_native_symbols_vec[i];
  235. if (node->module_name == module_name) {
  236. field_id = wasm_runtime_get_syssymbol_id(runtime, field_name);
  237. first_sym_id = wasm_runtime_get_syssymbol_id(runtime, node->native_symbols[0].u.symbol);
  238. if (field_id < first_sym_id + node->n_native_symbols) {
  239. symbol_index = field_id - first_sym_id;
  240. func_ptr = node->native_symbols[symbol_index].func_ptr;
  241. signature = node->native_symbols[symbol_index].signature;
  242. attachment = node->native_symbols[symbol_index].attachment;
  243. break;
  244. }
  245. }
  246. }
  247. } else {
  248. for (uint32 i = ID_USER; i < g_native_libs_count; i ++) {
  249. node = &g_native_symbols_vec[i];
  250. if (node->module_name == module_name) {
  251. func_ptr =
  252. lookup_symbol(node->native_symbols, node->n_native_symbols,
  253. field_name, &signature, &attachment);
  254. if (func_ptr) {
  255. break;
  256. }
  257. if (field_name->str[0] == '_') {
  258. const ConstStrDescription * new_field_name =
  259. wasm_runtime_records_const_string(runtime, &field_name->str[1], field_name->len - 1, NULL, 0);
  260. func_ptr =
  261. lookup_symbol(node->native_symbols, node->n_native_symbols,
  262. new_field_name, &signature, &attachment);
  263. if (func_ptr)
  264. break;
  265. }
  266. }
  267. }
  268. }
  269. #if 0
  270. node = g_native_symbols_list;
  271. while (node) {
  272. node_next = node->next;
  273. if (!strcmp(node->module_name, module_name)) {
  274. if ((func_ptr =
  275. lookup_symbol(node->native_symbols, node->n_native_symbols,
  276. field_name, &signature, &attachment))
  277. || (field_name[0] == '_'
  278. && (func_ptr = lookup_symbol(
  279. node->native_symbols, node->n_native_symbols,
  280. field_name + 1, &signature, &attachment))))
  281. break;
  282. }
  283. node = node_next;
  284. }
  285. #endif
  286. if (func_ptr) {
  287. if (signature && signature[0] != '\0') {
  288. /* signature is not empty, check its format */
  289. if (!check_symbol_signature(func_type, signature)) {
  290. // hotfix abort() if launching AS module
  291. if (symbol_index && node->native_symbols[++symbol_index].u.symbol == field_name) {
  292. signature = node->native_symbols[symbol_index].signature;
  293. if (check_symbol_signature(func_type, signature)) {
  294. *p_signature = signature;
  295. func_ptr = node->native_symbols[symbol_index].func_ptr;
  296. attachment = node->native_symbols[symbol_index].attachment;
  297. pass_check = true;
  298. }
  299. }
  300. if (!pass_check) {
  301. #if WASM_ENABLE_WAMR_COMPILER == 0
  302. /* Output warning except running aot compiler */
  303. LOG_WARNING("failed to check signature '%s' and resolve "
  304. "pointer params for import function (%s %s)\n",
  305. signature, module_name, field_name);
  306. #endif
  307. return NULL;
  308. }
  309. }
  310. else
  311. /* Save signature for runtime to do pointer check and
  312. address conversion */
  313. *p_signature = signature;
  314. }
  315. else
  316. /* signature is empty */
  317. *p_signature = NULL;
  318. *p_attachment = attachment;
  319. *p_call_conv_raw = node->call_conv_raw;
  320. }
  321. return func_ptr;
  322. }
  323. static bool
  324. register_natives(const char *module_name, NativeSymbol *native_symbols,
  325. uint32 n_native_symbols, bool call_conv_raw)
  326. {
  327. WASMRuntime * runtime = wasm_runtime_get_runtime();
  328. /*
  329. #if ENABLE_SORT_DEBUG != 0
  330. struct timeval start;
  331. struct timeval end;
  332. unsigned long timer;
  333. #endif
  334. */
  335. if (!runtime)
  336. return false;
  337. if ((g_native_libs_count + 1) > g_native_libs_size) {
  338. g_native_symbols_vec = wasm_runtime_realloc(
  339. g_native_symbols_vec, (g_native_libs_size + 2) * sizeof(NativeSymbolsNode));
  340. if (!g_native_symbols_vec) {
  341. return false;
  342. }
  343. g_native_libs_size += 2;
  344. bh_assert((g_native_libs_count + 2) == g_native_libs_size);
  345. }
  346. const ConstStrDescription * csp_module_name =
  347. wasm_runtime_records_const_string(runtime, module_name, strlen(module_name), NULL, 0);
  348. g_native_symbols_vec[g_native_libs_count].module_name = csp_module_name;
  349. g_native_symbols_vec[g_native_libs_count].native_symbols = NULL;
  350. g_native_symbols_vec[g_native_libs_count].n_native_symbols = n_native_symbols;
  351. g_native_symbols_vec[g_native_libs_count].call_conv_raw = call_conv_raw;
  352. NativeSymbol * csp_native_symbols = (NativeSymbol*)wasm_runtime_malloc(sizeof(NativeSymbol) * n_native_symbols);
  353. if (!csp_native_symbols) {
  354. return false;
  355. }
  356. for (uint32 i = 0; i < n_native_symbols; i++) {
  357. memcpy(&csp_native_symbols[i], &native_symbols[i], sizeof(NativeSymbol));
  358. csp_native_symbols[i].u.symbol =
  359. wasm_runtime_records_const_string(runtime, native_symbols[i].u.symbol_str,
  360. strlen(native_symbols[i].u.symbol_str), NULL, 0);
  361. }
  362. g_native_symbols_vec[g_native_libs_count].native_symbols = csp_native_symbols;
  363. g_native_libs_count ++;
  364. /*
  365. #if ENABLE_SORT_DEBUG != 0
  366. gettimeofday(&start, NULL);
  367. #endif
  368. #if ENABLE_QUICKSORT == 0
  369. sort_symbol_ptr(native_symbols, n_native_symbols);
  370. #else
  371. quick_sort_symbols(native_symbols, 0, (int)(n_native_symbols - 1));
  372. #endif
  373. #if ENABLE_SORT_DEBUG != 0
  374. gettimeofday(&end, NULL);
  375. timer =
  376. 1000000 * (end.tv_sec - start.tv_sec) + (end.tv_usec - start.tv_usec);
  377. LOG_ERROR("module_name: %s, nums: %d, sorted used: %ld us", module_name,
  378. n_native_symbols, timer);
  379. #endif
  380. */
  381. return true;
  382. }
  383. bool
  384. wasm_native_register_natives(const char *module_name,
  385. NativeSymbol *native_symbols,
  386. uint32 n_native_symbols)
  387. {
  388. return register_natives(module_name, native_symbols, n_native_symbols,
  389. false);
  390. }
  391. bool
  392. wasm_native_register_natives_raw(const char *module_name,
  393. NativeSymbol *native_symbols,
  394. uint32 n_native_symbols)
  395. {
  396. return register_natives(module_name, native_symbols, n_native_symbols,
  397. true);
  398. }
  399. bool
  400. wasm_native_init()
  401. {
  402. WASMRuntime * runtime = wasm_runtime_get_runtime();
  403. NativeSymbol *native_symbols;
  404. uint32 n_native_symbols = 0;
  405. uint32 n_native_nodes = g_native_libs_size = NODE_COUNT;
  406. g_native_libs_count = NODE_COUNT - 1;
  407. if (!runtime)
  408. return false;
  409. if (n_native_nodes) {
  410. g_native_symbols_vec = (NativeSymbolsNode*)wasm_runtime_malloc(
  411. sizeof(NativeSymbolsNode) * n_native_nodes);
  412. if (!g_native_symbols_vec)
  413. return false;
  414. memset(g_native_symbols_vec, 0, sizeof(NativeSymbolsNode) * n_native_nodes);
  415. }
  416. #if WASM_ENABLE_LIBC_BUILTIN != 0
  417. n_native_symbols = get_libc_builtin_export_apis(&native_symbols);
  418. //if (!wasm_native_register_natives(CONST_STR_POOL_DESC(WAMR_CSP_env), native_symbols, n_native_symbols))
  419. // return false;
  420. if (n_native_symbols) {
  421. for (uint32 i = 0; i < n_native_symbols; i ++)
  422. native_symbols[i].u.symbol = CONST_STR_POOL_DESC(runtime, native_symbols[i].u.symbol_key);
  423. g_native_symbols_vec[ID_LIBC_BUILTIN].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_env);
  424. g_native_symbols_vec[ID_LIBC_BUILTIN].native_symbols = native_symbols;
  425. g_native_symbols_vec[ID_LIBC_BUILTIN].n_native_symbols = n_native_symbols;
  426. g_native_symbols_vec[ID_LIBC_BUILTIN].call_conv_raw = false;
  427. }
  428. #endif
  429. #if WASM_ENABLE_SPEC_TEST != 0
  430. n_native_symbols = get_spectest_export_apis(&native_symbols);
  431. if (n_native_symbols) {
  432. for (uint32 i = 0; i < n_native_symbols; i ++)
  433. native_symbols[i].u.symbol = CONST_STR_POOL_DESC(runtime, native_symbols[i].u.symbol_key);
  434. g_native_symbols_vec[ID_SPECTEST].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_spectest);
  435. g_native_symbols_vec[ID_SPECTEST].native_symbols = native_symbols;
  436. g_native_symbols_vec[ID_SPECTEST].n_native_symbols = n_native_symbols;
  437. g_native_symbols_vec[ID_SPECTEST].call_conv_raw = false;
  438. }
  439. #endif
  440. #if WASM_ENABLE_LIBC_WASI != 0
  441. n_native_symbols = get_libc_wasi_export_apis(&native_symbols);
  442. if (n_native_symbols) {
  443. for (uint32 i = 0; i < n_native_symbols; i ++)
  444. native_symbols[i].u.symbol = CONST_STR_POOL_DESC(runtime, native_symbols[i].u.symbol_key);
  445. g_native_symbols_vec[ID_LIBC_WASI_UNSTABLE].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_wasi_unstable);
  446. g_native_symbols_vec[ID_LIBC_WASI_UNSTABLE].native_symbols = native_symbols;
  447. g_native_symbols_vec[ID_LIBC_WASI_UNSTABLE].n_native_symbols = n_native_symbols;
  448. g_native_symbols_vec[ID_LIBC_WASI_UNSTABLE].call_conv_raw = false;
  449. g_native_symbols_vec[ID_LIBC_WASI_PREVIEW1].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_wasi_snapshot_preview1);
  450. g_native_symbols_vec[ID_LIBC_WASI_PREVIEW1].native_symbols = native_symbols;
  451. g_native_symbols_vec[ID_LIBC_WASI_PREVIEW1].n_native_symbols = n_native_symbols;
  452. g_native_symbols_vec[ID_LIBC_WASI_PREVIEW1].call_conv_raw = false;
  453. }
  454. #endif
  455. #if WASM_ENABLE_BASE_LIB != 0
  456. n_native_symbols = get_base_lib_export_apis(&native_symbols);
  457. if (n_native_symbols) {
  458. for (uint32 i = 0; i < n_native_symbols; i ++)
  459. native_symbols[i].u.symbol = CONST_STR_POOL_DESC(runtime, native_symbols[i].u.symbol_key);
  460. g_native_symbols_vec[ID_BASE_LIB].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_env);
  461. g_native_symbols_vec[ID_BASE_LIB].native_symbols = native_symbols;
  462. g_native_symbols_vec[ID_BASE_LIB].n_native_symbols = n_native_symbols;
  463. g_native_symbols_vec[ID_BASE_LIB].call_conv_raw = false;
  464. }
  465. #endif
  466. #if WASM_ENABLE_APP_FRAMEWORK != 0
  467. n_native_symbols = get_ext_lib_export_apis(&native_symbols);
  468. if (n_native_symbols) {
  469. for (uint32 i = 0; i < n_native_symbols; i ++)
  470. native_symbols[i].u.symbol = CONST_STR_POOL_DESC(runtime, native_symbols[i].u.symbol_key);
  471. g_native_symbols_vec[ID_APP_FRAME].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_env);
  472. g_native_symbols_vec[ID_APP_FRAME].native_symbols = native_symbols;
  473. g_native_symbols_vec[ID_APP_FRAME].n_native_symbols = n_native_symbols;
  474. g_native_symbols_vec[ID_APP_FRAME].call_conv_raw = false;
  475. }
  476. #endif
  477. #if WASM_ENABLE_LIB_PTHREAD != 0
  478. if (!lib_pthread_init())
  479. return false;
  480. n_native_symbols = get_lib_pthread_export_apis(&native_symbols);
  481. if (n_native_symbols) {
  482. for (uint32 i = 0; i < n_native_symbols; i ++)
  483. native_symbols[i].u.symbol = CONST_STR_POOL_DESC(runtime, native_symbols[i].u.symbol_key);
  484. g_native_symbols_vec[ID_PTHREAD].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_env);
  485. g_native_symbols_vec[ID_PTHREAD].native_symbols = native_symbols;
  486. g_native_symbols_vec[ID_PTHREAD].n_native_symbols = n_native_symbols;
  487. g_native_symbols_vec[ID_PTHREAD].call_conv_raw = false;
  488. }
  489. #endif
  490. #if WASM_ENABLE_LIBC_EMCC != 0
  491. n_native_symbols = get_libc_emcc_export_apis(&native_symbols);
  492. if (n_native_symbols) {
  493. for (uint32 i = 0; i < n_native_symbols; i ++)
  494. native_symbols[i].u.symbol = CONST_STR_POOL_DESC(runtime, native_symbols[i].u.symbol_key);
  495. g_native_symbols_vec[ID_LIBC_EMCC].module_name = CONST_STR_POOL_DESC(runtime, WAMR_CSP_env);
  496. g_native_symbols_vec[ID_LIBC_EMCC].native_symbols = native_symbols;
  497. g_native_symbols_vec[ID_LIBC_EMCC].n_native_symbols = n_native_symbols;
  498. g_native_symbols_vec[ID_LIBC_EMCC].call_conv_raw = false;
  499. }
  500. #endif
  501. return true;
  502. }
  503. void
  504. wasm_native_destroy()
  505. {
  506. #if WASM_ENABLE_LIB_PTHREAD != 0
  507. lib_pthread_destroy();
  508. #endif
  509. for (uint32 i = ID_USER; i < g_native_libs_count; i ++) {
  510. if (g_native_symbols_vec[i].native_symbols)
  511. wasm_runtime_free(g_native_symbols_vec[i].native_symbols);
  512. }
  513. if (g_native_symbols_vec)
  514. wasm_runtime_free(g_native_symbols_vec);
  515. g_native_symbols_vec = NULL;
  516. }