| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 12:06:13 +01:00
										 |  |  | #include <LibCore/MemoryStream.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | #include <LibWasm/AbstractMachine/Configuration.h>
 | 
					
						
							|  |  |  | #include <LibWasm/AbstractMachine/Interpreter.h>
 | 
					
						
							| 
									
										
										
										
											2021-06-04 03:30:09 +04:30
										 |  |  | #include <LibWasm/Printer/Printer.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Wasm { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 02:55:01 +04:30
										 |  |  | Optional<size_t> Configuration::nth_label_index(size_t i) | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-18 00:21:00 +04:30
										 |  |  |     for (size_t index = m_stack.size(); index > 0; --index) { | 
					
						
							|  |  |  |         auto& entry = m_stack.entries()[index - 1]; | 
					
						
							| 
									
										
										
										
											2021-08-09 02:55:01 +04:30
										 |  |  |         if (entry.has<Label>()) { | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  |             if (i == 0) | 
					
						
							| 
									
										
										
										
											2021-08-09 02:55:01 +04:30
										 |  |  |                 return index - 1; | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  |             --i; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 03:42:11 +04:30
										 |  |  | void Configuration::unwind(Badge<CallFrameHandle>, CallFrameHandle const& frame_handle) | 
					
						
							| 
									
										
										
										
											2021-05-24 01:28:02 +04:30
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-04 03:30:09 +04:30
										 |  |  |     if (m_stack.size() == frame_handle.stack_size && frame_handle.frame_index == m_current_frame_index) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-24 01:28:02 +04:30
										 |  |  |     VERIFY(m_stack.size() > frame_handle.stack_size); | 
					
						
							|  |  |  |     m_stack.entries().remove(frame_handle.stack_size, m_stack.size() - frame_handle.stack_size); | 
					
						
							|  |  |  |     m_current_frame_index = frame_handle.frame_index; | 
					
						
							|  |  |  |     m_depth--; | 
					
						
							|  |  |  |     m_ip = frame_handle.ip; | 
					
						
							|  |  |  |     VERIFY(m_stack.size() == frame_handle.stack_size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-24 02:04:58 +04:30
										 |  |  | Result Configuration::call(Interpreter& interpreter, FunctionAddress address, Vector<Value> arguments) | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     auto* function = m_store.get(address); | 
					
						
							|  |  |  |     if (!function) | 
					
						
							|  |  |  |         return Trap {}; | 
					
						
							|  |  |  |     if (auto* wasm_function = function->get_pointer<WasmFunction>()) { | 
					
						
							| 
									
										
										
										
											2021-08-12 00:59:00 +04:30
										 |  |  |         Vector<Value> locals = move(arguments); | 
					
						
							|  |  |  |         locals.ensure_capacity(locals.size() + wasm_function->code().locals().size()); | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  |         for (auto& type : wasm_function->code().locals()) | 
					
						
							|  |  |  |             locals.empend(type, 0ull); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-23 21:42:19 +04:30
										 |  |  |         set_frame(Frame { | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  |             wasm_function->module(), | 
					
						
							|  |  |  |             move(locals), | 
					
						
							|  |  |  |             wasm_function->code().body(), | 
					
						
							| 
									
										
										
										
											2021-05-23 21:42:19 +04:30
										 |  |  |             wasm_function->type().results().size(), | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-05-24 01:28:02 +04:30
										 |  |  |         m_ip = 0; | 
					
						
							| 
									
										
										
										
											2021-05-24 02:04:58 +04:30
										 |  |  |         return execute(interpreter); | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // It better be a host function, else something is really wrong.
 | 
					
						
							|  |  |  |     auto& host_function = function->get<HostFunction>(); | 
					
						
							| 
									
										
										
										
											2021-05-17 12:29:44 +04:30
										 |  |  |     return host_function.function()(*this, arguments); | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-24 02:04:58 +04:30
										 |  |  | Result Configuration::execute(Interpreter& interpreter) | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     interpreter.interpret(*this); | 
					
						
							| 
									
										
										
										
											2021-05-07 09:59:18 +04:30
										 |  |  |     if (interpreter.did_trap()) | 
					
						
							| 
									
										
										
										
											2021-07-01 17:03:17 +04:30
										 |  |  |         return Trap { interpreter.trap_reason() }; | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 03:27:51 +04:30
										 |  |  |     if (stack().size() <= frame().arity() + 1) | 
					
						
							| 
									
										
										
										
											2021-07-01 17:03:17 +04:30
										 |  |  |         return Trap { "Not enough values to return from call" }; | 
					
						
							| 
									
										
										
										
											2021-06-04 03:27:51 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-23 21:21:17 +04:30
										 |  |  |     Vector<Value> results; | 
					
						
							| 
									
										
										
										
											2021-05-23 21:42:19 +04:30
										 |  |  |     results.ensure_capacity(frame().arity()); | 
					
						
							|  |  |  |     for (size_t i = 0; i < frame().arity(); ++i) | 
					
						
							| 
									
										
										
										
											2021-05-23 21:21:17 +04:30
										 |  |  |         results.append(move(stack().pop().get<Value>())); | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  |     auto label = stack().pop(); | 
					
						
							|  |  |  |     // ASSERT: label == current frame
 | 
					
						
							| 
									
										
										
										
											2021-05-23 21:21:17 +04:30
										 |  |  |     if (!label.has<Label>()) | 
					
						
							| 
									
										
										
										
											2021-07-01 17:03:17 +04:30
										 |  |  |         return Trap { "Invalid stack configuration" }; | 
					
						
							| 
									
										
										
										
											2021-05-23 21:21:17 +04:30
										 |  |  |     return Result { move(results) }; | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-01 03:19:01 +04:30
										 |  |  | void Configuration::dump_stack() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |     auto print_value = []<typename... Ts>(CheckedFormatString<Ts...> format, Ts... vs) { | 
					
						
							| 
									
										
										
										
											2023-01-09 12:06:13 +01:00
										 |  |  |         Core::Stream::AllocatingMemoryStream memory_stream; | 
					
						
							| 
									
										
										
										
											2023-01-21 10:29:21 +01:00
										 |  |  |         Printer { memory_stream }.print(vs...); | 
					
						
							| 
									
										
										
										
											2023-01-09 12:06:13 +01:00
										 |  |  |         auto buffer = ByteBuffer::create_uninitialized(memory_stream.used_buffer_size()).release_value_but_fixme_should_propagate_errors(); | 
					
						
							|  |  |  |         memory_stream.read_entire_buffer(buffer).release_value_but_fixme_should_propagate_errors(); | 
					
						
							| 
									
										
										
										
											2021-09-10 23:43:11 +02:00
										 |  |  |         dbgln(format.view(), StringView(buffer).trim_whitespace()); | 
					
						
							| 
									
										
										
										
											2021-06-04 03:30:09 +04:30
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2021-06-04 03:42:11 +04:30
										 |  |  |     for (auto const& entry : stack().entries()) { | 
					
						
							| 
									
										
										
										
											2021-05-01 03:19:01 +04:30
										 |  |  |         entry.visit( | 
					
						
							| 
									
										
										
										
											2021-06-04 03:42:11 +04:30
										 |  |  |             [&](Value const& v) { | 
					
						
							| 
									
										
										
										
											2021-06-04 03:30:09 +04:30
										 |  |  |                 print_value("    {}", v); | 
					
						
							| 
									
										
										
										
											2021-05-01 03:19:01 +04:30
										 |  |  |             }, | 
					
						
							| 
									
										
										
										
											2021-06-04 03:42:11 +04:30
										 |  |  |             [&](Frame const& f) { | 
					
						
							| 
									
										
										
										
											2021-05-23 21:42:19 +04:30
										 |  |  |                 dbgln("    frame({})", f.arity()); | 
					
						
							|  |  |  |                 for (auto& local : f.locals()) { | 
					
						
							| 
									
										
										
										
											2021-06-04 03:30:09 +04:30
										 |  |  |                     print_value("        {}", local); | 
					
						
							| 
									
										
										
										
											2021-05-01 03:19:01 +04:30
										 |  |  |                 } | 
					
						
							|  |  |  |             }, | 
					
						
							| 
									
										
										
										
											2021-06-04 03:42:11 +04:30
										 |  |  |             [](Label const& l) { | 
					
						
							| 
									
										
										
										
											2021-05-23 21:21:17 +04:30
										 |  |  |                 dbgln("    label({}) -> {}", l.arity(), l.continuation()); | 
					
						
							| 
									
										
										
										
											2021-05-01 03:19:01 +04:30
										 |  |  |             }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-01 01:08:51 +04:30
										 |  |  | } |