| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-11 13:02:42 +04:30
										 |  |  | #include <AK/StackInfo.h>
 | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  | #include <LibWasm/AbstractMachine/Configuration.h>
 | 
					
						
							|  |  |  | #include <LibWasm/AbstractMachine/Interpreter.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Wasm { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct BytecodeInterpreter : public Interpreter { | 
					
						
							| 
									
										
										
										
											2023-05-28 12:13:43 +02:00
										 |  |  |     explicit BytecodeInterpreter(StackInfo const& stack_info) | 
					
						
							|  |  |  |         : m_stack_info(stack_info) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  |     virtual void interpret(Configuration&) override; | 
					
						
							|  |  |  |     virtual ~BytecodeInterpreter() override = default; | 
					
						
							| 
									
										
										
										
											2023-02-25 11:15:11 +03:30
										 |  |  |     virtual bool did_trap() const override { return !m_trap.has<Empty>(); } | 
					
						
							|  |  |  |     virtual DeprecatedString trap_reason() const override | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_trap.visit( | 
					
						
							|  |  |  |             [](Empty) -> DeprecatedString { VERIFY_NOT_REACHED(); }, | 
					
						
							|  |  |  |             [](Trap const& trap) { return trap.reason; }, | 
					
						
							| 
									
										
										
										
											2023-08-09 08:49:02 +02:00
										 |  |  |             [](JS::Completion const& completion) { return completion.value()->to_string_without_side_effects().to_deprecated_string(); }); | 
					
						
							| 
									
										
										
										
											2023-02-25 11:15:11 +03:30
										 |  |  |     } | 
					
						
							|  |  |  |     virtual void clear_trap() override { m_trap = Empty {}; } | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     struct CallFrameHandle { | 
					
						
							|  |  |  |         explicit CallFrameHandle(BytecodeInterpreter& interpreter, Configuration& configuration) | 
					
						
							|  |  |  |             : m_configuration_handle(configuration) | 
					
						
							|  |  |  |             , m_interpreter(interpreter) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ~CallFrameHandle() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Configuration::CallFrameHandle m_configuration_handle; | 
					
						
							|  |  |  |         BytecodeInterpreter& m_interpreter; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     virtual void interpret(Configuration&, InstructionPointer&, Instruction const&); | 
					
						
							|  |  |  |     void branch_to_label(Configuration&, LabelIndex); | 
					
						
							|  |  |  |     template<typename ReadT, typename PushT> | 
					
						
							|  |  |  |     void load_and_push(Configuration&, Instruction const&); | 
					
						
							| 
									
										
										
										
											2021-08-11 22:16:05 +04:30
										 |  |  |     template<typename PopT, typename StoreT> | 
					
						
							|  |  |  |     void pop_and_store(Configuration&, Instruction const&); | 
					
						
							| 
									
										
										
										
											2021-12-04 17:56:58 +03:30
										 |  |  |     void store_to_memory(Configuration&, Instruction const&, ReadonlyBytes data, i32 base); | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  |     void call_address(Configuration&, FunctionAddress); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 02:55:01 +04:30
										 |  |  |     template<typename PopType, typename PushType, typename Operator> | 
					
						
							|  |  |  |     void binary_numeric_operation(Configuration&); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template<typename PopType, typename PushType, typename Operator> | 
					
						
							|  |  |  |     void unary_operation(Configuration&); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  |     template<typename V, typename T> | 
					
						
							|  |  |  |     MakeUnsigned<T> checked_unsigned_truncate(V); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template<typename V, typename T> | 
					
						
							|  |  |  |     MakeSigned<T> checked_signed_truncate(V); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template<typename T> | 
					
						
							|  |  |  |     T read_value(ReadonlyBytes data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Vector<Value> pop_values(Configuration& configuration, size_t count); | 
					
						
							| 
									
										
										
										
											2021-07-13 00:38:21 +04:30
										 |  |  |     ALWAYS_INLINE bool trap_if_not(bool value, StringView reason) | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  |     { | 
					
						
							|  |  |  |         if (!value) | 
					
						
							| 
									
										
										
										
											2021-07-06 14:28:55 +04:30
										 |  |  |             m_trap = Trap { reason }; | 
					
						
							| 
									
										
										
										
											2023-02-25 11:15:11 +03:30
										 |  |  |         return !m_trap.has<Empty>(); | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-01 17:03:17 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-25 11:15:11 +03:30
										 |  |  |     Variant<Trap, JS::Completion, Empty> m_trap; | 
					
						
							| 
									
										
										
										
											2023-05-28 12:13:43 +02:00
										 |  |  |     StackInfo const& m_stack_info; | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct DebuggerBytecodeInterpreter : public BytecodeInterpreter { | 
					
						
							| 
									
										
										
										
											2023-05-28 12:13:43 +02:00
										 |  |  |     DebuggerBytecodeInterpreter(StackInfo const& stack_info) | 
					
						
							|  |  |  |         : BytecodeInterpreter(stack_info) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-04 21:54:20 +10:00
										 |  |  |     virtual ~DebuggerBytecodeInterpreter() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Function<bool(Configuration&, InstructionPointer&, Instruction const&)> pre_interpret_hook; | 
					
						
							|  |  |  |     Function<bool(Configuration&, InstructionPointer&, Instruction const&, Interpreter const&)> post_interpret_hook; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     virtual void interpret(Configuration&, InstructionPointer&, Instruction const&) override; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |