mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-25 05:12:04 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			169 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			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.
 | |
| 
 | 
