fpc/packages/wasmtime/examples/linking.pp
Michaël Van Canneyt 0ec4203e2c * PChar -> PAnsiChar
2023-07-15 18:22:36 +02:00

169 lines
5.2 KiB
ObjectPascal

program linking;
uses classes,ctypes, wasmtime;
procedure exit_with_error(message : string; error : Pwasmtime_error_t; trap: Pwasm_trap_t); cdecl;
var
error_message : Twasm_byte_vec_t ;
S : AnsiString;
begin
Writeln(stderr, 'error: ', message);
S:='';
if (error <> Nil) then
begin
write('Error: ');
wasmtime_error_message(error, @error_message);
wasmtime_error_delete(error)
end
else
begin
write('Trap: ');
wasm_trap_message(trap, @error_message);
wasm_trap_delete(trap);
end;
SetLength(S,error_message.size);
Move(error_message.data^,S[1],error_message.size);
Writeln(stderr, '>>',S,'<<');
wasm_byte_vec_delete(@error_message);
halt(1);
end;
procedure checkerror(error : Pwasmtime_error_t; Message : string);
begin
if Assigned(error) then
exit_with_error(message,error,Nil);
end;
procedure checkerror(error : Pwasmtime_error_t; var trap : Pwasm_trap_t; Message : string);
begin
if Assigned(error) or assigned(Trap) then
exit_with_error(message,error,trap);
end;
Procedure read_wat_file(engine : Pwasm_engine_t; out bytes : twasm_byte_vec_t; aFile : String);
var
wat : twasm_byte_vec_t;
F : TMemoryStream;
begin
F:=TMemoryStream.Create;
try
F.LoadFromFile(aFile);
wasm_byte_vec_new_uninitialized(@wat, F.Size);
Move(F.Memory^,wat.data^,F.Size);
finally
F.Free;
end;
CheckError(wasmtime_wat2wasm(PAnsiChar(wat.data), wat.size, @bytes),
'failed to parse wat file '+aFile);
wasm_byte_vec_delete(@wat);
end;
Var
engine : Pwasm_engine_t = Nil;
store : Pwasmtime_store_t = Nil;
context : Pwasmtime_context_t = Nil;
linking1_wasm, linking2_wasm : twasm_byte_vec_t;
linking1_module,linking2_module : Pwasmtime_module_t;
error : Pwasmtime_error_t = Nil;
run : twasmtime_extern_t;
trap : Pwasm_trap_t = Nil;
linker : Pwasmtime_linker_t;
wasi_config : Pwasi_config_t;
status : cint;
linking1, linking2 : twasmtime_instance_t;
ok : Byte;
begin
linking1_module:=nil;
linking2_module:=Nil;
Writeln('Loading wasm library');
Loadwasmtime('./'+libwasmtime);
Writeln('Initializing...');
engine := wasm_engine_new();
store:=wasmtime_store_new(engine, nil,nil);
context:=wasmtime_store_context(store);
read_wat_file(engine, linking1_wasm, 'linking1.wat');
read_wat_file(engine, linking2_wasm, 'linking2.wat');
// Now that we've got our binary webassembly we can compile our module.
Writeln('Compiling module...');
error:=wasmtime_module_new(engine, Puint8_t(linking1_wasm.data), linking1_wasm.size, @linking1_module);
wasm_byte_vec_delete(@linking1_wasm);
if (error <> nil) then
exit_with_error('failed to compile module linking1', error, nil);
error:=wasmtime_module_new(engine, Puint8_t(linking2_wasm.data), linking2_wasm.size, @linking2_module);
wasm_byte_vec_delete(@linking2_wasm);
if (error <> nil) then
exit_with_error('failed to compile module linking2', error, nil);
Writeln('Configuring WASI...');
wasi_config:=wasi_config_new();
if (wasi_config=nil) then
exit_with_error('failed to create wasi config', Nil, nil);
wasi_config_inherit_argv(wasi_config);
wasi_config_inherit_env(wasi_config);
wasi_config_inherit_stdin(wasi_config);
wasi_config_inherit_stdout(wasi_config);
wasi_config_inherit_stderr(wasi_config);
wasi_config_preopen_dir(wasi_config,PAnsiChar('.'),PAnsiChar('.'));
CheckError(wasmtime_context_set_wasi(context, wasi_config),
'failed to instantiate WASI');
Writeln('Creating linker...');
linker:= wasmtime_linker_new(engine);
CheckError(wasmtime_linker_define_wasi(linker),'failed to define link wasi');
// Instantiate `linking2` with our linker.
CheckError(wasmtime_linker_instantiate(linker, context, linking2_module, @linking2, @trap),Trap,
'failed to instantiate linking2');
// Register our new `linking2` instance with the linker
CheckError(wasmtime_linker_define_instance(linker, context, PAnsiChar('linking2'), Length('linking2'), @linking2),
'failed to link linking2');
// Instantiate `linking1` with the linker now that `linking2` is defined
CheckError(wasmtime_linker_instantiate(linker, context, linking1_module, @linking1, @trap),trap,
'failed to instantiate linking1');
Writeln('Extracting export...');
ok:=wasmtime_instance_export_get(context, @linking1, PAnsiChar('run'), 3, @run) ;
if OK=0 then
exit_with_error('failed to get run export', nil, nil);
if run.kind<>WASMTIME_EXTERN_FUNC then
exit_with_error('run is not a function', nil, nil);
// And call it!
Writeln('Calling export...');
error:=wasmtime_func_call(context, @run.of_.func, nil, 0, nil, 0, @trap);
if (Trap<>Nil) then
begin
// exit_proc is reported as trap.
if wasmtime_trap_exit_status(trap,@status)<>0 then
Writeln('Wasm program exited with status: ',Status)
else
exit_with_error('failed to run default export for module', error, trap);
end
else if (error<>nil) then
exit_with_error('failed to run default export for module', error, trap);
// Clean up after ourselves at this point
Writeln('All finished!');
wasmtime_linker_delete(linker);
wasmtime_module_delete(linking1_module);
wasmtime_module_delete(linking2_module);
wasmtime_store_delete(store);
wasm_engine_delete(engine);
end.