mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-26 02:44:14 +00:00 
			
		
		
		
	 0e5ecef848
			
		
	
	
		0e5ecef848
		
	
	
	
	
		
			
			This commit adds a register allocator, with 8 available "register" slots. In testing with various random blobs, this moves anywhere from 30% to 74% of value accesses into predefined slots, and is about a ~20% perf increase end-to-end. To actually make this usable, a few structural changes were also made: - we no longer do one instruction per interpret call - trapping is an (unlikely) exit condition - the label and frame stacks are replaced with linked lists with a huge node cache size, as we only need to touch the last element and push/pop is very frequent.
		
			
				
	
	
		
			78 lines
		
	
	
	
		
			2.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			78 lines
		
	
	
	
		
			2.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/MemoryStream.h>
 | |
| #include <LibWasm/AbstractMachine/Configuration.h>
 | |
| #include <LibWasm/AbstractMachine/Interpreter.h>
 | |
| #include <LibWasm/Printer/Printer.h>
 | |
| 
 | |
| namespace Wasm {
 | |
| 
 | |
| void Configuration::unwind(Badge<CallFrameHandle>, CallFrameHandle const& frame_handle)
 | |
| {
 | |
|     m_frame_stack.take_last();
 | |
|     m_depth--;
 | |
|     m_ip = frame_handle.ip.value();
 | |
|     m_locals_base = m_frame_stack.is_empty() ? nullptr : m_frame_stack.unchecked_last().locals().data();
 | |
| }
 | |
| 
 | |
| Result Configuration::call(Interpreter& interpreter, FunctionAddress address, Vector<Value> arguments)
 | |
| {
 | |
|     auto* function = m_store.get(address);
 | |
|     if (!function)
 | |
|         return Trap::from_string("Attempt to call nonexistent function by address");
 | |
|     if (auto* wasm_function = function->get_pointer<WasmFunction>()) {
 | |
|         Vector<Value> locals = move(arguments);
 | |
|         locals.ensure_capacity(locals.size() + wasm_function->code().func().locals().size());
 | |
|         for (auto& local : wasm_function->code().func().locals()) {
 | |
|             for (size_t i = 0; i < local.n(); ++i)
 | |
|                 locals.append(Value(local.type()));
 | |
|         }
 | |
| 
 | |
|         set_frame(Frame {
 | |
|             wasm_function->module(),
 | |
|             move(locals),
 | |
|             wasm_function->code().func().body(),
 | |
|             wasm_function->type().results().size(),
 | |
|         });
 | |
|         m_ip = 0;
 | |
|         return execute(interpreter);
 | |
|     }
 | |
| 
 | |
|     // It better be a host function, else something is really wrong.
 | |
|     auto& host_function = function->get<HostFunction>();
 | |
|     return host_function.function()(*this, arguments);
 | |
| }
 | |
| 
 | |
| Result Configuration::execute(Interpreter& interpreter)
 | |
| {
 | |
|     interpreter.interpret(*this);
 | |
|     if (interpreter.did_trap())
 | |
|         return interpreter.trap();
 | |
| 
 | |
|     Vector<Value> results { value_stack().span().slice_from_end(frame().arity()) };
 | |
|     value_stack().shrink(value_stack().size() - results.size(), true);
 | |
|     results.reverse();
 | |
| 
 | |
|     label_stack().take_last();
 | |
|     return Result { move(results) };
 | |
| }
 | |
| 
 | |
| void Configuration::dump_stack()
 | |
| {
 | |
|     auto print_value = []<typename... Ts>(CheckedFormatString<Ts...> format, Ts... vs) {
 | |
|         AllocatingMemoryStream memory_stream;
 | |
|         Printer { memory_stream }.print(vs...);
 | |
|         auto buffer = ByteBuffer::create_uninitialized(memory_stream.used_buffer_size()).release_value_but_fixme_should_propagate_errors();
 | |
|         memory_stream.read_until_filled(buffer).release_value_but_fixme_should_propagate_errors();
 | |
|         dbgln(format.view(), StringView(buffer).trim_whitespace());
 | |
|     };
 | |
|     for (auto const& value : value_stack()) {
 | |
|         print_value("    {}", value);
 | |
|     }
 | |
| }
 | |
| 
 | |
| }
 |