wasmer_wasix/syscalls/wasix/
dlsym.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
use state::MAIN_MODULE_HANDLE;

use super::*;
use crate::{
    state::{ModuleHandle, ResolvedExport},
    syscalls::*,
};

#[instrument(level = "trace", skip_all, fields(symbol = field::Empty), ret)]
pub fn dlsym<M: MemorySize>(
    mut ctx: FunctionEnvMut<'_, WasiEnv>,
    handle: DlHandle,
    symbol: WasmPtr<u8, M>,
    symbol_len: M::Offset,
    err_buf: WasmPtr<u8, M>,
    err_buf_len: M::Offset,
    out_symbol: WasmPtr<M::Offset, M>,
) -> Result<Errno, WasiError> {
    WasiEnv::do_pending_operations(&mut ctx)?;

    let (env, mut store) = ctx.data_and_store_mut();
    let memory = unsafe { env.memory_view(&store) };
    let symbol = unsafe { get_input_str_ok!(&memory, symbol, symbol_len) };
    Span::current().record("symbol", symbol.as_str());

    let env_inner = unsafe { env.inner() };
    let Some(linker) = env_inner.linker() else {
        wasi_dl_err!(
            "The current instance is not a dynamically-linked instance",
            memory,
            err_buf,
            err_buf_len
        );
    };
    let linker = linker.clone();

    // handle = 0 is RTLD_DEFAULT, so search everywhere
    let handle = if handle == 0 {
        None
    } else {
        Some(ModuleHandle::from(handle))
    };
    let symbol = linker.resolve_export(&mut ctx, handle, &symbol);

    let (env, mut store) = ctx.data_and_store_mut();
    let memory = unsafe { env.memory_view(&store) };

    let symbol = wasi_try_dl!(
        symbol,
        "failed to resolve symbol: {}",
        memory,
        err_buf,
        err_buf_len
    );

    match symbol {
        ResolvedExport::Function { func_ptr: addr } | ResolvedExport::Global { data_ptr: addr } => {
            let Ok(addr) = addr.try_into() else {
                panic!("Failed to convert u64 address to M::Offset");
            };
            wasi_try_mem_ok!(out_symbol.write(&memory, addr));
        }
    }

    Ok(Errno::Success)
}