| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051 |
- /*
- * Copyright (C) 2024 Xiaomi Corporation. All rights reserved.
- * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- */
- #include "test_helper.h"
- #include "gtest/gtest.h"
- #include "bh_read_file.h"
- #include "wasm_runtime_common.h"
- #include <gtest/gtest-spi.h>
- class shared_heap_test : public testing::Test
- {
- protected:
- virtual void SetUp() {}
- static void SetUpTestCase() {}
- virtual void TearDown() {}
- WAMRRuntimeRAII<512 * 1024> runtime;
- };
- struct ret_env {
- wasm_exec_env_t exec_env;
- wasm_module_t wasm_module;
- wasm_module_inst_t wasm_module_inst;
- unsigned char *wasm_file_buf;
- char error_buf[128];
- };
- static void
- destroy_module_env(struct ret_env module_env);
- static bool
- load_wasm(char *wasm_file_tested, unsigned int app_heap_size,
- ret_env &ret_module_env)
- {
- char *wasm_file = strdup(wasm_file_tested);
- unsigned int wasm_file_size = 0;
- unsigned int stack_size = 16 * 1024, heap_size = app_heap_size;
- char error_buf[128] = { 0 };
- ret_module_env.wasm_file_buf =
- (unsigned char *)bh_read_file_to_buffer(wasm_file, &wasm_file_size);
- if (!ret_module_env.wasm_file_buf) {
- goto fail;
- }
- ret_module_env.wasm_module =
- wasm_runtime_load(ret_module_env.wasm_file_buf, wasm_file_size,
- error_buf, sizeof(error_buf));
- if (!ret_module_env.wasm_module) {
- memcpy(ret_module_env.error_buf, error_buf, 128);
- goto fail;
- }
- ret_module_env.wasm_module_inst =
- wasm_runtime_instantiate(ret_module_env.wasm_module, stack_size,
- heap_size, error_buf, sizeof(error_buf));
- if (!ret_module_env.wasm_module_inst) {
- memcpy(ret_module_env.error_buf, error_buf, 128);
- goto fail;
- }
- ret_module_env.exec_env = wasm_runtime_create_exec_env(
- ret_module_env.wasm_module_inst, stack_size);
- if (!ret_module_env.exec_env) {
- goto fail;
- }
- free(wasm_file);
- return true;
- fail:
- free(wasm_file);
- destroy_module_env(ret_module_env);
- return false;
- }
- void
- destroy_module_env(struct ret_env module_env)
- {
- if (module_env.exec_env) {
- wasm_runtime_destroy_exec_env(module_env.exec_env);
- }
- if (module_env.wasm_module_inst) {
- wasm_runtime_deinstantiate(module_env.wasm_module_inst);
- }
- if (module_env.wasm_module) {
- wasm_runtime_unload(module_env.wasm_module);
- }
- if (module_env.wasm_file_buf) {
- wasm_runtime_free(module_env.wasm_file_buf);
- }
- }
- static void
- test_shared_heap(WASMSharedHeap *shared_heap, const char *file,
- const char *func_name, uint32 argc, uint32 argv[])
- {
- struct ret_env tmp_module_env;
- WASMFunctionInstanceCommon *func_test = nullptr;
- bool ret = false;
- if (!load_wasm((char *)file, 0, tmp_module_env)) {
- ADD_FAILURE() << "Failed to load wasm file\n";
- goto fail0;
- }
- if (!wasm_runtime_attach_shared_heap(tmp_module_env.wasm_module_inst,
- shared_heap)) {
- ADD_FAILURE() << "Failed to attach shared heap\n";
- goto fail1;
- }
- func_test = wasm_runtime_lookup_function(tmp_module_env.wasm_module_inst,
- func_name);
- if (!func_test) {
- ADD_FAILURE() << "Failed to wasm_runtime_lookup_function!\n";
- goto fail2;
- }
- ret =
- wasm_runtime_call_wasm(tmp_module_env.exec_env, func_test, argc, argv);
- if (!ret) {
- const char *s =
- wasm_runtime_get_exception(tmp_module_env.wasm_module_inst);
- ADD_FAILURE() << "Failed to wasm_runtime_call_wasm with "
- << "exception: " << s;
- }
- fail2:
- wasm_runtime_detach_shared_heap(tmp_module_env.wasm_module_inst);
- fail1:
- destroy_module_env(tmp_module_env);
- fail0:
- return;
- }
- TEST_F(shared_heap_test, test_shared_heap_basic)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr;
- uint32 argv[1] = { 0 };
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- test_shared_heap(shared_heap, "test.wasm", "test", 0, argv);
- EXPECT_EQ(10, argv[0]);
- test_shared_heap(shared_heap, "test.aot", "test", 0, argv);
- EXPECT_EQ(10, argv[0]);
- test_shared_heap(shared_heap, "test_chain.aot", "test", 0, argv);
- EXPECT_EQ(10, argv[0]);
- }
- TEST_F(shared_heap_test, test_shared_heap_malloc_fail)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr;
- uint32 argv[1] = { 0 };
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- test_shared_heap(shared_heap, "test.wasm", "test_malloc_fail", 0, argv);
- EXPECT_EQ(1, argv[0]);
- test_shared_heap(shared_heap, "test.aot", "test_malloc_fail", 0, argv);
- EXPECT_EQ(1, argv[0]);
- test_shared_heap(shared_heap, "test_chain.aot", "test_malloc_fail", 0,
- argv);
- EXPECT_EQ(1, argv[0]);
- }
- TEST_F(shared_heap_test, test_preallocated_shared_heap_malloc_fail)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE];
- /* create a preallocated shared heap */
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- /* test wasm can't malloc with preallocated shared heap */
- argv[0] = 1024;
- test_shared_heap(shared_heap, "test.wasm", "my_shared_heap_malloc", 1,
- argv);
- EXPECT_EQ(0, argv[0]);
- argv[0] = 1024;
- test_shared_heap(shared_heap, "test.aot", "my_shared_heap_malloc", 1, argv);
- EXPECT_EQ(0, argv[0]);
- argv[0] = 1024;
- test_shared_heap(shared_heap, "test_chain.aot", "my_shared_heap_malloc", 1,
- argv);
- EXPECT_EQ(0, argv[0]);
- }
- static void
- create_test_shared_heap(uint8 *preallocated_buf, size_t size,
- WASMSharedHeap **shared_heap_res)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr;
- args.pre_allocated_addr = preallocated_buf;
- args.size = size;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- *shared_heap_res = shared_heap;
- if (!*shared_heap_res) {
- FAIL() << "Create shared heap chain failed.\n";
- }
- }
- static void
- create_test_shared_heap_chain(uint8 *preallocated_buf, size_t size,
- uint8 *preallocated_buf2, size_t size2,
- WASMSharedHeap **shared_heap_chain)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr;
- args.pre_allocated_addr = preallocated_buf;
- args.size = size;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf2;
- args.size = size2;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- *shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- if (!*shared_heap_chain) {
- FAIL() << "Create shared heap chain failed.\n";
- }
- }
- TEST_F(shared_heap_test, test_shared_heap_rmw)
- {
- WASMSharedHeap *shared_heap = nullptr;
- uint32 argv[2] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE] = { 0 };
- uint32 start1, end1;
- create_test_shared_heap(preallocated_buf, BUF_SIZE, &shared_heap);
- /* app addr for shared heap */
- start1 = UINT32_MAX - BUF_SIZE + 1;
- end1 = UINT32_MAX;
- argv[0] = end1;
- argv[1] = 101;
- test_shared_heap(shared_heap, "test.wasm", "read_modify_write_8", 2, argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf[BUF_SIZE - 1], 101);
- argv[0] = start1;
- argv[1] = 37;
- test_shared_heap(shared_heap, "test.wasm", "read_modify_write_8", 2, argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf[0], 37);
- argv[0] = end1;
- argv[1] = 81;
- test_shared_heap(shared_heap, "test.aot", "read_modify_write_8", 2, argv);
- EXPECT_EQ(101, argv[0]);
- EXPECT_EQ(preallocated_buf[BUF_SIZE - 1], 81);
- argv[0] = start1;
- argv[1] = 98;
- test_shared_heap(shared_heap, "test.aot", "read_modify_write_8", 2, argv);
- EXPECT_EQ(37, argv[0]);
- EXPECT_EQ(preallocated_buf[0], 98);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_rmw)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap_chain = nullptr;
- uint32 argv[2] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE] = { 0 },
- preallocated_buf2[BUF_SIZE] = { 0 };
- uint32 start1, end1, start2, end2;
- create_test_shared_heap_chain(preallocated_buf, BUF_SIZE, preallocated_buf2,
- BUF_SIZE, &shared_heap_chain);
- /* app addr for shared heap */
- start1 = UINT32_MAX - 2 * BUF_SIZE + 1;
- end1 = UINT32_MAX - BUF_SIZE;
- start2 = UINT32_MAX - BUF_SIZE + 1;
- end2 = UINT32_MAX;
- /* shared heap 1 */
- argv[0] = end1;
- argv[1] = 101;
- test_shared_heap(shared_heap_chain, "test.wasm", "read_modify_write_8", 2,
- argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf[BUF_SIZE - 1], 101);
- /* shared heap 2 */
- argv[0] = start2;
- argv[1] = 129;
- test_shared_heap(shared_heap_chain, "test.wasm", "read_modify_write_8", 2,
- argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf2[0], 129);
- argv[0] = start1;
- argv[1] = 98;
- test_shared_heap(shared_heap_chain, "test_chain.aot", "read_modify_write_8",
- 2, argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf[0], 98);
- argv[0] = end2;
- argv[1] = 81;
- test_shared_heap(shared_heap_chain, "test_chain.aot", "read_modify_write_8",
- 2, argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf2[BUF_SIZE - 1], 81);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_rmw_bulk_memory)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap_chain = nullptr;
- uint32 argv[3] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE] = { 0 },
- preallocated_buf2[BUF_SIZE] = { 0 };
- uint32 start1, end1, start2, end2;
- create_test_shared_heap_chain(preallocated_buf, BUF_SIZE, preallocated_buf2,
- BUF_SIZE, &shared_heap_chain);
- /* app addr for shared heap */
- start1 = UINT32_MAX - 2 * BUF_SIZE + 1;
- end1 = UINT32_MAX - BUF_SIZE;
- start2 = UINT32_MAX - BUF_SIZE + 1;
- end2 = UINT32_MAX;
- argv[0] = end1;
- argv[1] = 101;
- argv[2] = 1;
- test_shared_heap(shared_heap_chain, "test_bulk_memory.wasm",
- "memory_fill_test", 3, argv);
- /* no modification since no return value */
- EXPECT_EQ(end1, argv[0]);
- EXPECT_EQ(preallocated_buf[BUF_SIZE - 1], 101);
- argv[0] = start1;
- argv[1] = 14;
- argv[2] = 1;
- test_shared_heap(shared_heap_chain, "test_bulk_memory_chain.aot",
- "memory_fill_test", 3, argv);
- /* no modification since no return value */
- EXPECT_EQ(start1, argv[0]);
- EXPECT_EQ(preallocated_buf[0], 14);
- /* nothing happen when memory fill 0 byte */
- argv[0] = start2;
- argv[1] = 68;
- argv[2] = 0;
- test_shared_heap(shared_heap_chain, "test_bulk_memory_chain.aot",
- "memory_fill_test", 3, argv);
- /* no modification since no return value */
- EXPECT_EQ(start2, argv[0]);
- EXPECT_EQ(preallocated_buf2[0], 0);
- argv[0] = end2;
- argv[1] = 98;
- argv[2] = 1;
- test_shared_heap(shared_heap_chain, "test_bulk_memory_chain.aot",
- "memory_fill_test", 3, argv);
- /* no modification since no return value */
- EXPECT_EQ(end2, argv[0]);
- EXPECT_EQ(preallocated_buf2[BUF_SIZE - 1], 98);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_rmw_bulk_memory_oob)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap_chain = nullptr;
- uint32 argv[3] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE] = { 0 },
- preallocated_buf2[BUF_SIZE] = { 0 };
- uint32 start1, end1, start2, end2;
- create_test_shared_heap_chain(preallocated_buf, BUF_SIZE, preallocated_buf2,
- BUF_SIZE, &shared_heap_chain);
- /* app addr for shared heap */
- start1 = UINT32_MAX - 2 * BUF_SIZE + 1;
- end1 = UINT32_MAX - BUF_SIZE;
- start2 = UINT32_MAX - BUF_SIZE + 1;
- end2 = UINT32_MAX;
- /* shared heap 1 */
- argv[0] = end1;
- argv[1] = 101;
- argv[2] = 2;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory.wasm",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- argv[0] = end2;
- argv[1] = 98;
- argv[2] = 2;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory.wasm",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- argv[0] = start1;
- argv[1] = 98;
- argv[2] = BUF_SIZE + 1;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory.wasm",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- argv[0] = start2;
- argv[1] = 98;
- argv[2] = BUF_SIZE + 1;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory.wasm",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- argv[0] = end1;
- argv[1] = 101;
- argv[2] = 2;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory_chain.aot",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- argv[0] = end2;
- argv[1] = 98;
- argv[2] = 2;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory_chain.aot",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- argv[0] = start1;
- argv[1] = 98;
- argv[2] = BUF_SIZE + 1;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory_chain.aot",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- argv[0] = start2;
- argv[1] = 98;
- argv[2] = BUF_SIZE + 1;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_bulk_memory_chain.aot",
- "memory_fill_test", 3, argv),
- "Exception: out of bounds memory access");
- }
- TEST_F(shared_heap_test, test_shared_heap_rmw_oob)
- {
- WASMSharedHeap *shared_heap = nullptr;
- uint32 argv[2] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE], preallocated_buf2[BUF_SIZE];
- uint32 start1, end1, start2, end2;
- create_test_shared_heap(preallocated_buf, BUF_SIZE, &shared_heap);
- /* app addr for shared heap */
- start1 = UINT32_MAX - BUF_SIZE + 1;
- end1 = UINT32_MAX;
- /* try to rmw an u16, first u8 is in the first shared heap and second u8 is
- * in the second shared heap, will be seen as oob */
- argv[0] = end1;
- argv[1] = 12025;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap, "test.wasm",
- "read_modify_write_16", 2, argv),
- "Exception: out of bounds memory access");
- argv[0] = start1 - 1;
- argv[1] = 12025;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap, "test.aot",
- "read_modify_write_16", 2, argv),
- "Exception: out of bounds memory access");
- argv[0] = end1;
- argv[1] = 12025;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap, "test.aot",
- "read_modify_write_16", 2, argv),
- "Exception: out of bounds memory access");
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_rmw_oob)
- {
- WASMSharedHeap *shared_heap_chain = nullptr;
- uint32 argv[2] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE], preallocated_buf2[BUF_SIZE];
- uint32 start1, end1, start2, end2;
- create_test_shared_heap_chain(preallocated_buf, BUF_SIZE, preallocated_buf2,
- BUF_SIZE, &shared_heap_chain);
- /* app addr for shared heap */
- start1 = UINT32_MAX - 2 * BUF_SIZE + 1;
- end1 = UINT32_MAX - BUF_SIZE;
- start2 = UINT32_MAX - BUF_SIZE + 1;
- end2 = UINT32_MAX;
- /* try to rmw an u16, first u8 is in the first shared heap and second u8 is
- * in the second shared heap, will be seen as oob */
- argv[0] = end2;
- argv[1] = 12025;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain, "test.wasm",
- "read_modify_write_16", 2, argv),
- "Exception: out of bounds memory access");
- argv[0] = end1;
- argv[1] = 12025;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_chain.aot",
- "read_modify_write_16", 2, argv),
- "Exception: out of bounds memory access");
- }
- #if WASM_ENABLE_MEMORY64 != 0
- TEST_F(shared_heap_test, test_shared_heap_chain_memory64_rmw)
- {
- WASMSharedHeap *shared_heap_chain = nullptr;
- uint32 argv[3] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE] = { 0 },
- preallocated_buf2[BUF_SIZE] = { 0 };
- uint64 start1, end1, start2, end2;
- create_test_shared_heap_chain(preallocated_buf, BUF_SIZE, preallocated_buf2,
- BUF_SIZE, &shared_heap_chain);
- /* app addr for shared heap */
- start1 = UINT64_MAX - 2 * BUF_SIZE + 1;
- end1 = UINT64_MAX - BUF_SIZE;
- start2 = UINT64_MAX - BUF_SIZE + 1;
- end2 = UINT64_MAX;
- /* shared heap 1 */
- PUT_I64_TO_ADDR(argv, end1);
- argv[2] = 101;
- test_shared_heap(shared_heap_chain, "test64.wasm", "read_modify_write_8", 3,
- argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf[BUF_SIZE - 1], 101);
- /* shared heap 2 */
- PUT_I64_TO_ADDR(argv, start2);
- argv[2] = 129;
- test_shared_heap(shared_heap_chain, "test64.wasm", "read_modify_write_8", 3,
- argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf2[0], 129);
- PUT_I64_TO_ADDR(argv, start1);
- argv[2] = 98;
- test_shared_heap(shared_heap_chain, "test64_chain.aot",
- "read_modify_write_8", 3, argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf[0], 98);
- PUT_I64_TO_ADDR(argv, end2);
- argv[2] = 81;
- test_shared_heap(shared_heap_chain, "test64_chain.aot",
- "read_modify_write_8", 3, argv);
- EXPECT_EQ(0, argv[0]);
- EXPECT_EQ(preallocated_buf2[BUF_SIZE - 1], 81);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_memory64_rmw_oob)
- {
- WASMSharedHeap *shared_heap_chain = nullptr;
- uint32 argv[3] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE], preallocated_buf2[BUF_SIZE];
- uint64 start1, end1, start2, end2;
- create_test_shared_heap_chain(preallocated_buf, BUF_SIZE, preallocated_buf2,
- BUF_SIZE, &shared_heap_chain);
- /* app addr for shared heap */
- start1 = UINT64_MAX - 2 * BUF_SIZE + 1;
- end1 = UINT64_MAX - BUF_SIZE;
- start2 = UINT64_MAX - BUF_SIZE + 1;
- end2 = UINT64_MAX;
- /* try to rmw an u16, first u8 is in the first shared heap and second u8 is
- * in the second shared heap, will be seen as oob */
- PUT_I64_TO_ADDR(argv, end1);
- argv[2] = 12025;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain, "test64.wasm",
- "read_modify_write_16", 3, argv),
- "Exception: out of bounds memory access");
- PUT_I64_TO_ADDR(argv, end1);
- argv[2] = 12025;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test64_chain.aot",
- "read_modify_write_16", 3, argv),
- "Exception: out of bounds memory access");
- }
- #endif
- #ifndef native_function
- /* clang-format off */
- #define native_function(func_name, signature) \
- { #func_name, (void *)glue_## func_name, signature, NULL }
- /* clang-format on */
- #endif
- #ifndef nitems
- #define nitems(_a) (sizeof(_a) / sizeof(0 [(_a)]))
- #endif /* nitems */
- uintptr_t
- glue_test_addr_conv(wasm_exec_env_t env, uintptr_t addr)
- {
- wasm_module_inst_t module_inst = get_module_inst(env);
- void *native_addr = (void *)addr;
- uintptr_t app_addr = addr_native_to_app(native_addr);
- native_addr = addr_app_to_native(app_addr);
- if (native_addr != (void *)addr) {
- ADD_FAILURE() << "address conversion incorrect";
- return 0;
- }
- return app_addr;
- }
- static NativeSymbol g_test_native_symbols[] = {
- native_function(test_addr_conv, "(*)i"),
- };
- TEST_F(shared_heap_test, test_addr_conv)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr;
- uint32 argv[1] = { 0 };
- bool ret = false;
- ret = wasm_native_register_natives("env", g_test_native_symbols,
- nitems(g_test_native_symbols));
- if (!ret) {
- FAIL() << "Failed to register natives";
- }
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- test_shared_heap(shared_heap, "test_addr_conv.wasm", "test", 0, argv);
- EXPECT_EQ(1, argv[0]);
- test_shared_heap(shared_heap, "test_addr_conv.aot", "test", 0, argv);
- EXPECT_EQ(1, argv[0]);
- test_shared_heap(shared_heap, "test_addr_conv_chain.aot", "test", 0, argv);
- EXPECT_EQ(1, argv[0]);
- }
- TEST_F(shared_heap_test, test_addr_conv_pre_allocated_oob)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize(),
- app_addr = 0xFFFFFFFF - BUF_SIZE;
- uint8 preallocated_buf[BUF_SIZE];
- bool ret = false;
- /* create a preallocated shared heap */
- ret = wasm_native_register_natives("env", g_test_native_symbols,
- nitems(g_test_native_symbols));
- if (!ret) {
- FAIL() << "Failed to register natives";
- }
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- argv[0] = app_addr;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap, "test_addr_conv.wasm",
- "test_preallocated", 1, argv),
- "Exception: out of bounds memory access");
- argv[0] = app_addr;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap, "test_addr_conv.aot",
- "test_preallocated", 1, argv),
- "Exception: out of bounds memory access");
- argv[0] = app_addr;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap,
- "test_addr_conv_chain.aot",
- "test_preallocated", 1, argv),
- "Exception: out of bounds memory access");
- }
- TEST_F(shared_heap_test, test_shared_heap_chain)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr,
- *shared_heap_chain = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE];
- bool ret = false;
- ret = wasm_native_register_natives("env", g_test_native_symbols,
- nitems(g_test_native_symbols));
- if (!ret) {
- FAIL() << "Failed to register natives";
- }
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- /* create a preallocated shared heap */
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- if (!shared_heap_chain) {
- FAIL() << "Create shared heap chain failed.\n";
- }
- test_shared_heap(shared_heap_chain, "test_addr_conv.wasm", "test", 0, argv);
- EXPECT_EQ(1, argv[0]);
- test_shared_heap(shared_heap, "test_addr_conv.aot", "test", 0, argv);
- EXPECT_EQ(1, argv[0]);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_create_fail)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr,
- *shared_heap_chain = nullptr;
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- args.size = 4096;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- EXPECT_EQ(shared_heap_chain, nullptr);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_create_fail2)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr,
- *shared_heap_chain = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE];
- struct ret_env tmp_module_env;
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- if (!load_wasm((char *)"test.wasm", 0, tmp_module_env)) {
- FAIL() << "Failed to load wasm file\n";
- }
- if (!wasm_runtime_attach_shared_heap(tmp_module_env.wasm_module_inst,
- shared_heap)) {
- FAIL() << "Failed to attach shared heap\n";
- }
- /* can't create shared heap chain when shared heap is attached to a wasm
- * app */
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- EXPECT_EQ(shared_heap_chain, nullptr);
- wasm_runtime_detach_shared_heap(tmp_module_env.wasm_module_inst);
- destroy_module_env(tmp_module_env);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_create_fail3)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr,
- *shared_heap3 = nullptr, *shared_heap_chain = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE], preallocated_buf2[BUF_SIZE];
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- if (!shared_heap_chain) {
- FAIL() << "Create shared heap chain failed.\n";
- }
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf2;
- args.size = BUF_SIZE;
- shared_heap3 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap3) {
- FAIL() << "Failed to create shared heap";
- }
- /* The head and body can't be already in other shared heap chain as body */
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap3, shared_heap2);
- EXPECT_EQ(shared_heap_chain, nullptr);
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap2, shared_heap);
- EXPECT_EQ(shared_heap_chain, nullptr);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_unchain)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr,
- *shared_heap3 = nullptr, *shared_heap_chain = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE], preallocated_buf2[BUF_SIZE];
- args.size = 1024;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- if (!shared_heap_chain) {
- FAIL() << "Create shared heap chain failed.\n";
- }
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf2;
- args.size = BUF_SIZE;
- shared_heap3 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap3) {
- FAIL() << "Failed to create shared heap";
- }
- /* unchain shared heap so that the 'body' can be another chain 'body'
- * again(1->2 to 1->3->2) */
- EXPECT_EQ(shared_heap2,
- wasm_runtime_unchain_shared_heaps(shared_heap_chain, false));
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap3, shared_heap2);
- EXPECT_EQ(shared_heap_chain, shared_heap3);
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap3);
- EXPECT_EQ(shared_heap, shared_heap_chain);
- /* break down the entire shared heap chain */
- EXPECT_EQ(shared_heap2,
- wasm_runtime_unchain_shared_heaps(shared_heap_chain, true));
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_addr_conv)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr,
- *shared_heap_chain = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE];
- bool ret = false;
- ret = wasm_native_register_natives("env", g_test_native_symbols,
- nitems(g_test_native_symbols));
- if (!ret) {
- FAIL() << "Failed to register natives";
- }
- args.size = 4096;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- /* create a preallocated shared heap */
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- if (!shared_heap_chain) {
- FAIL() << "Create shared heap chain failed.\n";
- }
- argv[0] = 0xFFFFFFFF;
- test_shared_heap(shared_heap_chain, "test_addr_conv.wasm",
- "test_preallocated", 1, argv);
- EXPECT_EQ(1, argv[0]);
- argv[0] = 0xFFFFF000;
- test_shared_heap(shared_heap_chain, "test_addr_conv.wasm",
- "test_preallocated", 1, argv);
- EXPECT_EQ(1, argv[0]);
- argv[0] = 0xFFFFFFFF;
- test_shared_heap(shared_heap, "test_addr_conv_chain.aot",
- "test_preallocated", 1, argv);
- EXPECT_EQ(1, argv[0]);
- argv[0] = 0xFFFFF000;
- test_shared_heap(shared_heap, "test_addr_conv_chain.aot",
- "test_preallocated", 1, argv);
- EXPECT_EQ(1, argv[0]);
- }
- TEST_F(shared_heap_test, test_shared_heap_chain_addr_conv_oob)
- {
- SharedHeapInitArgs args = { 0 };
- WASMSharedHeap *shared_heap = nullptr, *shared_heap2 = nullptr,
- *shared_heap_chain = nullptr;
- uint32 argv[1] = { 0 }, BUF_SIZE = os_getpagesize();
- uint8 preallocated_buf[BUF_SIZE];
- bool ret = false;
- ret = wasm_native_register_natives("env", g_test_native_symbols,
- nitems(g_test_native_symbols));
- if (!ret) {
- FAIL() << "Failed to register natives";
- }
- args.size = 4096;
- shared_heap = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap) {
- FAIL() << "Failed to create shared heap";
- }
- /* create a preallocated shared heap */
- memset(&args, 0, sizeof(args));
- args.pre_allocated_addr = preallocated_buf;
- args.size = BUF_SIZE;
- shared_heap2 = wasm_runtime_create_shared_heap(&args);
- if (!shared_heap2) {
- FAIL() << "Create preallocated shared heap failed.\n";
- }
- shared_heap_chain =
- wasm_runtime_chain_shared_heaps(shared_heap, shared_heap2);
- if (!shared_heap_chain) {
- FAIL() << "Create shared heap chain failed.\n";
- }
- /* test wasm */
- argv[0] = 0xFFFFFFFF - BUF_SIZE - 4096;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_addr_conv.wasm",
- "test_preallocated", 1, argv),
- "Exception: out of bounds memory access");
- /* test aot */
- argv[0] = 0xFFFFFFFF - BUF_SIZE - 4096;
- EXPECT_NONFATAL_FAILURE(test_shared_heap(shared_heap_chain,
- "test_addr_conv_chain.aot",
- "test_preallocated", 1, argv),
- "Exception: out of bounds memory access");
- }
|