| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165 |
- /*
- * Copyright (C) 2019 Intel Corporation. All rights reserved.
- * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- */
- #include "uvwasi.h"
- #include "bh_platform.h"
- #include "wasm_export.h"
- /* clang-format off */
- #define get_module_inst(exec_env) \
- wasm_runtime_get_module_inst(exec_env)
- #define validate_app_addr(offset, size) \
- wasm_runtime_validate_app_addr(module_inst, offset, size)
- #define validate_native_addr(addr, size) \
- wasm_runtime_validate_native_addr(module_inst, addr, size)
- #define addr_app_to_native(offset) \
- wasm_runtime_addr_app_to_native(module_inst, offset)
- #define addr_native_to_app(ptr) \
- wasm_runtime_addr_native_to_app(module_inst, ptr)
- #define module_malloc(size, p_native_addr) \
- wasm_runtime_module_malloc(module_inst, size, p_native_addr)
- #define module_free(offset) \
- wasm_runtime_module_free(module_inst, offset)
- /* clang-format on */
- // uvwasi_errno_t is typedef'd to uint16 which is correct according to the ABI
- // specification. However, in WASM, the smallest integer type is int32. If we
- // return uint16, we would rely on language SDKs to implement the correct
- // behaviour of casting to uint16 before checking the value or using it any way.
- // Failure to do so can cause tricky bugs as the upper 16 bits of the error
- // result are not guaranteed to be zero'ed by us so the result essentially
- // contains garbage from the WASM app perspective. To prevent this, we return
- // uint32 directly instead so as not to be reliant on the correct behaviour of
- // any current/future SDK implementations.
- #define wasi_errno_t uint32_t
- #define wasi_fd_t uvwasi_fd_t
- #define wasi_clockid_t uvwasi_clockid_t
- #define wasi_timestamp_t uvwasi_timestamp_t
- #define wasi_filesize_t uvwasi_filesize_t
- #define wasi_prestat_app_t uvwasi_prestat_app_t
- #define wasi_filedelta_t uvwasi_filedelta_t
- #define wasi_whence_t uvwasi_whence_t
- #define wasi_fdflags_t uvwasi_fdflags_t
- #define wasi_rights_t uvwasi_rights_t
- #define wasi_advice_t uvwasi_advice_t
- #define wasi_lookupflags_t uvwasi_lookupflags_t
- #define wasi_preopentype_t uvwasi_preopentype_t
- #define wasi_fdstat_t uvwasi_fdstat_t
- #define wasi_oflags_t uvwasi_oflags_t
- #define wasi_dircookie_t uvwasi_dircookie_t
- #define wasi_filestat_t uvwasi_filestat_t
- #define wasi_fstflags_t uvwasi_fstflags_t
- #define wasi_subscription_t uvwasi_subscription_t
- #define wasi_event_t uvwasi_event_t
- #define wasi_exitcode_t uvwasi_exitcode_t
- #define wasi_signal_t uvwasi_signal_t
- #define wasi_riflags_t uvwasi_riflags_t
- #define wasi_roflags_t uvwasi_roflags_t
- #define wasi_siflags_t uvwasi_siflags_t
- #define wasi_sdflags_t uvwasi_sdflags_t
- #define wasi_iovec_t uvwasi_iovec_t
- #define wasi_ciovec_t uvwasi_ciovec_t
- typedef struct wasi_prestat_app {
- wasi_preopentype_t pr_type;
- uint32 pr_name_len;
- } wasi_prestat_app_t;
- typedef struct iovec_app {
- uint32 buf_offset;
- uint32 buf_len;
- } iovec_app_t;
- typedef struct WASIContext {
- uvwasi_t uvwasi;
- uint32_t exit_code;
- } WASIContext;
- void *
- wasm_runtime_get_wasi_ctx(wasm_module_inst_t module_inst);
- static uvwasi_t *
- get_wasi_ctx(wasm_module_inst_t module_inst)
- {
- WASIContext *ctx = wasm_runtime_get_wasi_ctx(module_inst);
- if (ctx == NULL) {
- return NULL;
- }
- return &ctx->uvwasi;
- }
- static wasi_errno_t
- wasi_args_get(wasm_exec_env_t exec_env, uint32 *argv_offsets, char *argv_buf)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_size_t argc, argv_buf_size, i;
- char **argv;
- uint64 total_size;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- err = uvwasi_args_sizes_get(uvwasi, &argc, &argv_buf_size);
- if (err)
- return err;
- total_size = sizeof(int32) * ((uint64)argc + 1);
- if (total_size >= UINT32_MAX
- || !validate_native_addr(argv_offsets, total_size)
- || argv_buf_size >= UINT32_MAX
- || !validate_native_addr(argv_buf, (uint64)argv_buf_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(char *) * ((uint64)argc + 1);
- if (total_size >= UINT32_MAX
- || !(argv = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- err = uvwasi_args_get(uvwasi, argv, argv_buf);
- if (err) {
- wasm_runtime_free(argv);
- return err;
- }
- for (i = 0; i < argc; i++)
- argv_offsets[i] = (uint32)addr_native_to_app(argv[i]);
- wasm_runtime_free(argv);
- return 0;
- }
- static wasi_errno_t
- wasi_args_sizes_get(wasm_exec_env_t exec_env, uint32 *argc_app,
- uint32 *argv_buf_size_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_size_t argc, argv_buf_size;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(argc_app, (uint64)sizeof(uint32))
- || !validate_native_addr(argv_buf_size_app, (uint64)sizeof(uint32)))
- return (wasi_errno_t)-1;
- err = uvwasi_args_sizes_get(uvwasi, &argc, &argv_buf_size);
- if (err)
- return err;
- *argc_app = (uint32)argc;
- *argv_buf_size_app = (uint32)argv_buf_size;
- return 0;
- }
- static wasi_errno_t
- wasi_clock_res_get(wasm_exec_env_t exec_env, wasi_clockid_t clock_id,
- wasi_timestamp_t *resolution)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!validate_native_addr(resolution, (uint64)sizeof(wasi_timestamp_t)))
- return (wasi_errno_t)-1;
- return uvwasi_clock_res_get(uvwasi, clock_id, resolution);
- }
- static wasi_errno_t
- wasi_clock_time_get(wasm_exec_env_t exec_env, wasi_clockid_t clock_id,
- wasi_timestamp_t precision, wasi_timestamp_t *time)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!validate_native_addr(time, (uint64)sizeof(wasi_timestamp_t)))
- return (wasi_errno_t)-1;
- return uvwasi_clock_time_get(uvwasi, clock_id, precision, time);
- }
- static wasi_errno_t
- wasi_environ_get(wasm_exec_env_t exec_env, uint32 *environ_offsets,
- char *environ_buf)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_size_t environ_count, environ_buf_size, i;
- uint64 total_size;
- char **environs;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- err = uvwasi_environ_sizes_get(uvwasi, &environ_count, &environ_buf_size);
- if (err)
- return err;
- if (environ_count == 0)
- return 0;
- total_size = sizeof(int32) * ((uint64)environ_count + 1);
- if (total_size >= UINT32_MAX
- || !validate_native_addr(environ_offsets, total_size)
- || environ_buf_size >= UINT32_MAX
- || !validate_native_addr(environ_buf, (uint64)environ_buf_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(char *) * (((uint64)environ_count + 1));
- if (total_size >= UINT32_MAX
- || !(environs = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- err = uvwasi_environ_get(uvwasi, environs, environ_buf);
- if (err) {
- wasm_runtime_free(environs);
- return err;
- }
- for (i = 0; i < environ_count; i++)
- environ_offsets[i] = (uint32)addr_native_to_app(environs[i]);
- wasm_runtime_free(environs);
- return 0;
- }
- static wasi_errno_t
- wasi_environ_sizes_get(wasm_exec_env_t exec_env, uint32 *environ_count_app,
- uint32 *environ_buf_size_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_size_t environ_count, environ_buf_size;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(environ_count_app, (uint64)sizeof(uint32))
- || !validate_native_addr(environ_buf_size_app, (uint64)sizeof(uint32)))
- return (wasi_errno_t)-1;
- err = uvwasi_environ_sizes_get(uvwasi, &environ_count, &environ_buf_size);
- if (err)
- return err;
- *environ_count_app = (uint32)environ_count;
- *environ_buf_size_app = (uint32)environ_buf_size;
- return 0;
- }
- static wasi_errno_t
- wasi_fd_prestat_get(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_prestat_app_t *prestat_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_prestat_t prestat;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(prestat_app, (uint64)sizeof(wasi_prestat_app_t)))
- return (wasi_errno_t)-1;
- err = uvwasi_fd_prestat_get(uvwasi, fd, &prestat);
- if (err)
- return err;
- prestat_app->pr_type = prestat.pr_type;
- prestat_app->pr_name_len = (uint32)prestat.u.dir.pr_name_len;
- return 0;
- }
- static wasi_errno_t
- wasi_fd_prestat_dir_name(wasm_exec_env_t exec_env, wasi_fd_t fd, char *path,
- uint32 path_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_prestat_dir_name(uvwasi, fd, path, path_len);
- }
- static wasi_errno_t
- wasi_fd_close(wasm_exec_env_t exec_env, wasi_fd_t fd)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_close(uvwasi, fd);
- }
- static wasi_errno_t
- wasi_fd_datasync(wasm_exec_env_t exec_env, wasi_fd_t fd)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_datasync(uvwasi, fd);
- }
- static wasi_errno_t
- wasi_fd_pread(wasm_exec_env_t exec_env, wasi_fd_t fd, iovec_app_t *iovec_app,
- uint32 iovs_len, wasi_filesize_t offset, uint32 *nread_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_iovec_t *iovec, *iovec_begin;
- uint64 total_size;
- uvwasi_size_t nread;
- uint32 i;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- total_size = sizeof(iovec_app_t) * (uint64)iovs_len;
- if (!validate_native_addr(nread_app, (uint64)sizeof(uint32))
- || total_size >= UINT32_MAX
- || !validate_native_addr(iovec_app, total_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(wasi_iovec_t) * (uint64)iovs_len;
- if (total_size >= UINT32_MAX
- || !(iovec_begin = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- iovec = iovec_begin;
- for (i = 0; i < iovs_len; i++, iovec_app++, iovec++) {
- if (!validate_app_addr((uint64)iovec_app->buf_offset,
- (uint64)iovec_app->buf_len)) {
- err = (wasi_errno_t)-1;
- goto fail;
- }
- iovec->buf = (void *)addr_app_to_native((uint64)iovec_app->buf_offset);
- iovec->buf_len = iovec_app->buf_len;
- }
- err = uvwasi_fd_pread(uvwasi, fd, iovec_begin, iovs_len, offset, &nread);
- if (err)
- goto fail;
- *nread_app = (uint32)nread;
- /* success */
- err = 0;
- fail:
- wasm_runtime_free(iovec_begin);
- return err;
- }
- static wasi_errno_t
- wasi_fd_pwrite(wasm_exec_env_t exec_env, wasi_fd_t fd,
- const iovec_app_t *iovec_app, uint32 iovs_len,
- wasi_filesize_t offset, uint32 *nwritten_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_ciovec_t *ciovec, *ciovec_begin;
- uint64 total_size;
- uvwasi_size_t nwritten;
- uint32 i;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- total_size = sizeof(iovec_app_t) * (uint64)iovs_len;
- if (!validate_native_addr(nwritten_app, (uint64)sizeof(uint32))
- || total_size >= UINT32_MAX
- || !validate_native_addr((void *)iovec_app, total_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(wasi_ciovec_t) * (uint64)iovs_len;
- if (total_size >= UINT32_MAX
- || !(ciovec_begin = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- ciovec = ciovec_begin;
- for (i = 0; i < iovs_len; i++, iovec_app++, ciovec++) {
- if (!validate_app_addr((uint64)iovec_app->buf_offset,
- (uint64)iovec_app->buf_len)) {
- err = (wasi_errno_t)-1;
- goto fail;
- }
- ciovec->buf = (char *)addr_app_to_native((uint64)iovec_app->buf_offset);
- ciovec->buf_len = iovec_app->buf_len;
- }
- err =
- uvwasi_fd_pwrite(uvwasi, fd, ciovec_begin, iovs_len, offset, &nwritten);
- if (err)
- goto fail;
- *nwritten_app = (uint32)nwritten;
- /* success */
- err = 0;
- fail:
- wasm_runtime_free(ciovec_begin);
- return err;
- }
- static wasi_errno_t
- wasi_fd_read(wasm_exec_env_t exec_env, wasi_fd_t fd,
- const iovec_app_t *iovec_app, uint32 iovs_len, uint32 *nread_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_iovec_t *iovec, *iovec_begin;
- uint64 total_size;
- uvwasi_size_t nread;
- uint32 i;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- total_size = sizeof(iovec_app_t) * (uint64)iovs_len;
- if (!validate_native_addr(nread_app, (uint64)sizeof(uint32))
- || total_size >= UINT32_MAX
- || !validate_native_addr((void *)iovec_app, total_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(wasi_iovec_t) * (uint64)iovs_len;
- if (total_size >= UINT32_MAX
- || !(iovec_begin = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- iovec = iovec_begin;
- for (i = 0; i < iovs_len; i++, iovec_app++, iovec++) {
- if (!validate_app_addr((uint64)iovec_app->buf_offset,
- (uint64)iovec_app->buf_len)) {
- err = (wasi_errno_t)-1;
- goto fail;
- }
- iovec->buf = (void *)addr_app_to_native((uint64)iovec_app->buf_offset);
- iovec->buf_len = iovec_app->buf_len;
- }
- err = uvwasi_fd_read(uvwasi, fd, iovec_begin, iovs_len, &nread);
- if (err)
- goto fail;
- *nread_app = (uint32)nread;
- /* success */
- err = 0;
- fail:
- wasm_runtime_free(iovec_begin);
- return err;
- }
- static wasi_errno_t
- wasi_fd_renumber(wasm_exec_env_t exec_env, wasi_fd_t from, wasi_fd_t to)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_renumber(uvwasi, from, to);
- }
- static wasi_errno_t
- wasi_fd_seek(wasm_exec_env_t exec_env, wasi_fd_t fd, wasi_filedelta_t offset,
- wasi_whence_t whence, wasi_filesize_t *newoffset)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(newoffset, (uint64)sizeof(wasi_filesize_t)))
- return (wasi_errno_t)-1;
- return uvwasi_fd_seek(uvwasi, fd, offset, whence, newoffset);
- }
- static wasi_errno_t
- wasi_fd_tell(wasm_exec_env_t exec_env, wasi_fd_t fd, wasi_filesize_t *newoffset)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(newoffset, (uint64)sizeof(wasi_filesize_t)))
- return (wasi_errno_t)-1;
- return uvwasi_fd_tell(uvwasi, fd, newoffset);
- }
- static wasi_errno_t
- wasi_fd_fdstat_get(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_fdstat_t *fdstat_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_fdstat_t fdstat;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(fdstat_app, (uint64)sizeof(wasi_fdstat_t)))
- return (wasi_errno_t)-1;
- err = uvwasi_fd_fdstat_get(uvwasi, fd, &fdstat);
- if (err)
- return err;
- memcpy(fdstat_app, &fdstat, sizeof(wasi_fdstat_t));
- return 0;
- }
- static wasi_errno_t
- wasi_fd_fdstat_set_flags(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_fdflags_t flags)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_fdstat_set_flags(uvwasi, fd, flags);
- }
- static wasi_errno_t
- wasi_fd_fdstat_set_rights(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_rights_t fs_rights_base,
- wasi_rights_t fs_rights_inheriting)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_fdstat_set_rights(uvwasi, fd, fs_rights_base,
- fs_rights_inheriting);
- }
- static wasi_errno_t
- wasi_fd_sync(wasm_exec_env_t exec_env, wasi_fd_t fd)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_sync(uvwasi, fd);
- }
- static wasi_errno_t
- wasi_fd_write(wasm_exec_env_t exec_env, wasi_fd_t fd,
- const iovec_app_t *iovec_app, uint32 iovs_len,
- uint32 *nwritten_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_ciovec_t *ciovec, *ciovec_begin;
- uint64 total_size;
- uvwasi_size_t nwritten;
- uint32 i;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- total_size = sizeof(iovec_app_t) * (uint64)iovs_len;
- if (!validate_native_addr(nwritten_app, (uint64)sizeof(uint32))
- || total_size >= UINT32_MAX
- || !validate_native_addr((void *)iovec_app, total_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(wasi_ciovec_t) * (uint64)iovs_len;
- if (total_size >= UINT32_MAX
- || !(ciovec_begin = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- ciovec = ciovec_begin;
- for (i = 0; i < iovs_len; i++, iovec_app++, ciovec++) {
- if (!validate_app_addr((uint64)iovec_app->buf_offset,
- (uint64)iovec_app->buf_len)) {
- err = (wasi_errno_t)-1;
- goto fail;
- }
- ciovec->buf = (char *)addr_app_to_native((uint64)iovec_app->buf_offset);
- ciovec->buf_len = iovec_app->buf_len;
- }
- #ifndef BH_VPRINTF
- err = uvwasi_fd_write(uvwasi, fd, ciovec_begin, iovs_len, &nwritten);
- #else
- /* redirect stdout/stderr output to BH_VPRINTF function */
- if (fd == 1 || fd == 2) {
- int i;
- const struct iovec *iov1 = (const struct iovec *)ciovec_begin;
- nwritten = 0;
- for (i = 0; i < (int)iovs_len; i++, iov1++) {
- if (iov1->iov_len > 0 && iov1->iov_base) {
- char format[16];
- /* make up format string "%.ns" */
- snprintf(format, sizeof(format), "%%.%ds", (int)iov1->iov_len);
- nwritten += (uvwasi_size_t)os_printf(format, iov1->iov_base);
- }
- }
- err = 0;
- }
- else {
- err = uvwasi_fd_write(uvwasi, fd, ciovec_begin, iovs_len, &nwritten);
- }
- #endif /* end of BH_VPRINTF */
- if (err)
- goto fail;
- *nwritten_app = (uint32)nwritten;
- /* success */
- err = 0;
- fail:
- wasm_runtime_free(ciovec_begin);
- return err;
- }
- static wasi_errno_t
- wasi_fd_advise(wasm_exec_env_t exec_env, wasi_fd_t fd, wasi_filesize_t offset,
- wasi_filesize_t len, wasi_advice_t advice)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_advise(uvwasi, fd, offset, len, advice);
- }
- static wasi_errno_t
- wasi_fd_allocate(wasm_exec_env_t exec_env, wasi_fd_t fd, wasi_filesize_t offset,
- wasi_filesize_t len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_allocate(uvwasi, fd, offset, len);
- }
- static wasi_errno_t
- wasi_path_create_directory(wasm_exec_env_t exec_env, wasi_fd_t fd,
- const char *path, uint32 path_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_path_create_directory(uvwasi, fd, path, path_len);
- }
- static wasi_errno_t
- wasi_path_link(wasm_exec_env_t exec_env, wasi_fd_t old_fd,
- wasi_lookupflags_t old_flags, const char *old_path,
- uint32 old_path_len, wasi_fd_t new_fd, const char *new_path,
- uint32 new_path_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_path_link(uvwasi, old_fd, old_flags, old_path, old_path_len,
- new_fd, new_path, new_path_len);
- }
- static wasi_errno_t
- wasi_path_open(wasm_exec_env_t exec_env, wasi_fd_t dirfd,
- wasi_lookupflags_t dirflags, const char *path, uint32 path_len,
- wasi_oflags_t oflags, wasi_rights_t fs_rights_base,
- wasi_rights_t fs_rights_inheriting, wasi_fdflags_t fs_flags,
- wasi_fd_t *fd_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_fd_t fd = (wasi_fd_t)-1; /* set fd_app -1 if path open failed */
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(fd_app, (uint64)sizeof(wasi_fd_t)))
- return (wasi_errno_t)-1;
- err = uvwasi_path_open(uvwasi, dirfd, dirflags, path, path_len, oflags,
- fs_rights_base, fs_rights_inheriting, fs_flags, &fd);
- *fd_app = fd;
- return err;
- }
- static wasi_errno_t
- wasi_fd_readdir(wasm_exec_env_t exec_env, wasi_fd_t fd, void *buf,
- uint32 buf_len, wasi_dircookie_t cookie, uint32 *bufused_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_size_t bufused;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(bufused_app, (uint64)sizeof(uint32)))
- return (wasi_errno_t)-1;
- err = uvwasi_fd_readdir(uvwasi, fd, buf, buf_len, cookie, &bufused);
- if (err)
- return err;
- *bufused_app = (uint32)bufused;
- return 0;
- }
- static wasi_errno_t
- wasi_path_readlink(wasm_exec_env_t exec_env, wasi_fd_t fd, const char *path,
- uint32 path_len, char *buf, uint32 buf_len,
- uint32 *bufused_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_size_t bufused;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(bufused_app, (uint64)sizeof(uint32)))
- return (wasi_errno_t)-1;
- err = uvwasi_path_readlink(uvwasi, fd, path, path_len, buf, buf_len,
- &bufused);
- if (err)
- return err;
- *bufused_app = (uint32)bufused;
- return 0;
- }
- static wasi_errno_t
- wasi_path_rename(wasm_exec_env_t exec_env, wasi_fd_t old_fd,
- const char *old_path, uint32 old_path_len, wasi_fd_t new_fd,
- const char *new_path, uint32 new_path_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_path_rename(uvwasi, old_fd, old_path, old_path_len, new_fd,
- new_path, new_path_len);
- }
- static wasi_errno_t
- wasi_fd_filestat_get(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_filestat_t *filestat)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(filestat, (uint64)sizeof(wasi_filestat_t)))
- return (wasi_errno_t)-1;
- return uvwasi_fd_filestat_get(uvwasi, fd, filestat);
- }
- static wasi_errno_t
- wasi_fd_filestat_set_times(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_timestamp_t st_atim, wasi_timestamp_t st_mtim,
- wasi_fstflags_t fstflags)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_filestat_set_times(uvwasi, fd, st_atim, st_mtim, fstflags);
- }
- static wasi_errno_t
- wasi_fd_filestat_set_size(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_filesize_t st_size)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_fd_filestat_set_size(uvwasi, fd, st_size);
- }
- static wasi_errno_t
- wasi_path_filestat_get(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_lookupflags_t flags, const char *path,
- uint32 path_len, wasi_filestat_t *filestat)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr(filestat, (uint64)sizeof(wasi_filestat_t)))
- return (wasi_errno_t)-1;
- return uvwasi_path_filestat_get(uvwasi, fd, flags, path, path_len,
- filestat);
- }
- static wasi_errno_t
- wasi_path_filestat_set_times(wasm_exec_env_t exec_env, wasi_fd_t fd,
- wasi_lookupflags_t flags, const char *path,
- uint32 path_len, wasi_timestamp_t st_atim,
- wasi_timestamp_t st_mtim, wasi_fstflags_t fstflags)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_path_filestat_set_times(uvwasi, fd, flags, path, path_len,
- st_atim, st_mtim, fstflags);
- }
- static wasi_errno_t
- wasi_path_symlink(wasm_exec_env_t exec_env, const char *old_path,
- uint32 old_path_len, wasi_fd_t fd, const char *new_path,
- uint32 new_path_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_path_symlink(uvwasi, old_path, old_path_len, fd, new_path,
- new_path_len);
- }
- static wasi_errno_t
- wasi_path_unlink_file(wasm_exec_env_t exec_env, wasi_fd_t fd, const char *path,
- uint32 path_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_path_unlink_file(uvwasi, fd, path, path_len);
- }
- static wasi_errno_t
- wasi_path_remove_directory(wasm_exec_env_t exec_env, wasi_fd_t fd,
- const char *path, uint32 path_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_path_remove_directory(uvwasi, fd, path, path_len);
- }
- static wasi_errno_t
- wasi_poll_oneoff(wasm_exec_env_t exec_env, const wasi_subscription_t *in,
- wasi_event_t *out, uint32 nsubscriptions, uint32 *nevents_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- uvwasi_size_t nevents;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- if (!validate_native_addr((void *)in, (uint64)sizeof(wasi_subscription_t))
- || !validate_native_addr(out, (uint64)sizeof(wasi_event_t))
- || !validate_native_addr(nevents_app, (uint64)sizeof(uint32)))
- return (wasi_errno_t)-1;
- err = uvwasi_poll_oneoff(uvwasi, in, out, nsubscriptions, &nevents);
- if (err)
- return err;
- *nevents_app = (uint32)nevents;
- return 0;
- }
- static void
- wasi_proc_exit(wasm_exec_env_t exec_env, wasi_exitcode_t rval)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- WASIContext *wasi_ctx = wasm_runtime_get_wasi_ctx(module_inst);
- /* Here throwing exception is just to let wasm app exit,
- the upper layer should clear the exception and return
- as normal */
- wasm_runtime_set_exception(module_inst, "wasi proc exit");
- wasi_ctx->exit_code = rval;
- }
- static wasi_errno_t
- wasi_proc_raise(wasm_exec_env_t exec_env, wasi_signal_t sig)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- char buf[32];
- snprintf(buf, sizeof(buf), "%s%d", "wasi proc raise ", sig);
- wasm_runtime_set_exception(module_inst, buf);
- return 0;
- }
- static wasi_errno_t
- wasi_random_get(wasm_exec_env_t exec_env, void *buf, uint32 buf_len)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- return uvwasi_random_get(uvwasi, buf, buf_len);
- }
- static wasi_errno_t
- wasi_sock_recv(wasm_exec_env_t exec_env, wasi_fd_t sock, iovec_app_t *ri_data,
- uint32 ri_data_len, wasi_riflags_t ri_flags,
- uint32 *ro_datalen_app, wasi_roflags_t *ro_flags)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_iovec_t *iovec, *iovec_begin;
- uint64 total_size;
- uvwasi_size_t ro_datalen;
- uint32 i;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- total_size = sizeof(iovec_app_t) * (uint64)ri_data_len;
- if (!validate_native_addr(ro_datalen_app, (uint32)sizeof(uint32))
- || !validate_native_addr(ro_flags, (uint32)sizeof(wasi_roflags_t))
- || total_size >= UINT32_MAX
- || !validate_native_addr(ri_data, (uint32)total_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(wasi_iovec_t) * (uint64)ri_data_len;
- if (total_size >= UINT32_MAX
- || !(iovec_begin = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- iovec = iovec_begin;
- for (i = 0; i < ri_data_len; i++, ri_data++, iovec++) {
- if (!validate_app_addr((uint64)ri_data->buf_offset,
- (uint64)ri_data->buf_len)) {
- err = (wasi_errno_t)-1;
- goto fail;
- }
- iovec->buf = (void *)addr_app_to_native((uint64)ri_data->buf_offset);
- iovec->buf_len = ri_data->buf_len;
- }
- err = uvwasi_sock_recv(uvwasi, sock, iovec_begin, ri_data_len, ri_flags,
- &ro_datalen, ro_flags);
- if (err)
- goto fail;
- *(uint32 *)ro_datalen_app = (uint32)ro_datalen;
- /* success */
- err = 0;
- fail:
- wasm_runtime_free(iovec_begin);
- return err;
- }
- static wasi_errno_t
- wasi_sock_send(wasm_exec_env_t exec_env, wasi_fd_t sock,
- const iovec_app_t *si_data, uint32 si_data_len,
- wasi_siflags_t si_flags, uint32 *so_datalen_app)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- wasi_ciovec_t *ciovec, *ciovec_begin;
- uint64 total_size;
- uvwasi_size_t so_datalen;
- uint32 i;
- wasi_errno_t err;
- if (!uvwasi)
- return (wasi_errno_t)-1;
- total_size = sizeof(iovec_app_t) * (uint64)si_data_len;
- if (!validate_native_addr(so_datalen_app, (uint64)sizeof(uint32))
- || total_size >= UINT32_MAX
- || !validate_native_addr((void *)si_data, total_size))
- return (wasi_errno_t)-1;
- total_size = sizeof(wasi_ciovec_t) * (uint64)si_data_len;
- if (total_size >= UINT32_MAX
- || !(ciovec_begin = wasm_runtime_malloc((uint32)total_size)))
- return (wasi_errno_t)-1;
- ciovec = ciovec_begin;
- for (i = 0; i < si_data_len; i++, si_data++, ciovec++) {
- if (!validate_app_addr((uint64)si_data->buf_offset,
- (uint64)si_data->buf_len)) {
- err = (wasi_errno_t)-1;
- goto fail;
- }
- ciovec->buf = (char *)addr_app_to_native((uint64)si_data->buf_offset);
- ciovec->buf_len = si_data->buf_len;
- }
- err = uvwasi_sock_send(uvwasi, sock, ciovec_begin, si_data_len, si_flags,
- &so_datalen);
- if (err)
- goto fail;
- *so_datalen_app = (uint32)so_datalen;
- /* success */
- err = 0;
- fail:
- wasm_runtime_free(ciovec_begin);
- return err;
- }
- static wasi_errno_t
- wasi_sock_shutdown(wasm_exec_env_t exec_env, wasi_fd_t sock, wasi_sdflags_t how)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- if (!uvwasi)
- return (wasi_errno_t)-1;
- return uvwasi_sock_shutdown(uvwasi, sock, how);
- }
- static wasi_errno_t
- wasi_sched_yield(wasm_exec_env_t exec_env)
- {
- wasm_module_inst_t module_inst = get_module_inst(exec_env);
- uvwasi_t *uvwasi = get_wasi_ctx(module_inst);
- return uvwasi_sched_yield(uvwasi);
- }
- /* clang-format off */
- #define REG_NATIVE_FUNC(func_name, signature) \
- { #func_name, wasi_##func_name, signature, NULL }
- /* clang-format on */
- static NativeSymbol native_symbols_libc_wasi[] = {
- REG_NATIVE_FUNC(args_get, "(**)i"),
- REG_NATIVE_FUNC(args_sizes_get, "(**)i"),
- REG_NATIVE_FUNC(clock_res_get, "(i*)i"),
- REG_NATIVE_FUNC(clock_time_get, "(iI*)i"),
- REG_NATIVE_FUNC(environ_get, "(**)i"),
- REG_NATIVE_FUNC(environ_sizes_get, "(**)i"),
- REG_NATIVE_FUNC(fd_prestat_get, "(i*)i"),
- REG_NATIVE_FUNC(fd_prestat_dir_name, "(i*~)i"),
- REG_NATIVE_FUNC(fd_close, "(i)i"),
- REG_NATIVE_FUNC(fd_datasync, "(i)i"),
- REG_NATIVE_FUNC(fd_pread, "(i*iI*)i"),
- REG_NATIVE_FUNC(fd_pwrite, "(i*iI*)i"),
- REG_NATIVE_FUNC(fd_read, "(i*i*)i"),
- REG_NATIVE_FUNC(fd_renumber, "(ii)i"),
- REG_NATIVE_FUNC(fd_seek, "(iIi*)i"),
- REG_NATIVE_FUNC(fd_tell, "(i*)i"),
- REG_NATIVE_FUNC(fd_fdstat_get, "(i*)i"),
- REG_NATIVE_FUNC(fd_fdstat_set_flags, "(ii)i"),
- REG_NATIVE_FUNC(fd_fdstat_set_rights, "(iII)i"),
- REG_NATIVE_FUNC(fd_sync, "(i)i"),
- REG_NATIVE_FUNC(fd_write, "(i*i*)i"),
- REG_NATIVE_FUNC(fd_advise, "(iIIi)i"),
- REG_NATIVE_FUNC(fd_allocate, "(iII)i"),
- REG_NATIVE_FUNC(path_create_directory, "(i*~)i"),
- REG_NATIVE_FUNC(path_link, "(ii*~i*~)i"),
- REG_NATIVE_FUNC(path_open, "(ii*~iIIi*)i"),
- REG_NATIVE_FUNC(fd_readdir, "(i*~I*)i"),
- REG_NATIVE_FUNC(path_readlink, "(i*~*~*)i"),
- REG_NATIVE_FUNC(path_rename, "(i*~i*~)i"),
- REG_NATIVE_FUNC(fd_filestat_get, "(i*)i"),
- REG_NATIVE_FUNC(fd_filestat_set_times, "(iIIi)i"),
- REG_NATIVE_FUNC(fd_filestat_set_size, "(iI)i"),
- REG_NATIVE_FUNC(path_filestat_get, "(ii*~*)i"),
- REG_NATIVE_FUNC(path_filestat_set_times, "(ii*~IIi)i"),
- REG_NATIVE_FUNC(path_symlink, "(*~i*~)i"),
- REG_NATIVE_FUNC(path_unlink_file, "(i*~)i"),
- REG_NATIVE_FUNC(path_remove_directory, "(i*~)i"),
- REG_NATIVE_FUNC(poll_oneoff, "(**i*)i"),
- REG_NATIVE_FUNC(proc_exit, "(i)"),
- REG_NATIVE_FUNC(proc_raise, "(i)i"),
- REG_NATIVE_FUNC(random_get, "(*~)i"),
- REG_NATIVE_FUNC(sock_recv, "(i*ii**)i"),
- REG_NATIVE_FUNC(sock_send, "(i*ii*)i"),
- REG_NATIVE_FUNC(sock_shutdown, "(ii)i"),
- REG_NATIVE_FUNC(sched_yield, "()i"),
- };
- uint32
- get_libc_wasi_export_apis(NativeSymbol **p_libc_wasi_apis)
- {
- *p_libc_wasi_apis = native_symbols_libc_wasi;
- return sizeof(native_symbols_libc_wasi) / sizeof(NativeSymbol);
- }
|