123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168 |
- 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.
|