Index: src/interpreter/setup-interpreter-internal.cc |
diff --git a/src/interpreter/setup-interpreter-internal.cc b/src/interpreter/setup-interpreter-internal.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..1477ee8c7b145250c99e6ac3c1a66abf73cf0888 |
--- /dev/null |
+++ b/src/interpreter/setup-interpreter-internal.cc |
@@ -0,0 +1,94 @@ |
+// Copyright 2017 the V8 project authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "src/interpreter/setup-interpreter.h" |
+ |
+#include "src/handles-inl.h" |
+#include "src/interpreter/bytecodes.h" |
+#include "src/interpreter/interpreter-generator.h" |
+#include "src/interpreter/interpreter.h" |
+#include "src/objects-inl.h" |
+ |
+namespace v8 { |
+namespace internal { |
+namespace interpreter { |
+ |
+// static |
+void SetupInterpreter::InstallBytecodeHandlers(Interpreter* interpreter) { |
+ DCHECK(!interpreter->IsDispatchTableInitialized()); |
+ HandleScope scope(interpreter->isolate_); |
+ Address* dispatch_table = interpreter->dispatch_table_; |
+ |
+ // Generate bytecode handlers for all bytecodes and scales. |
+ const OperandScale kOperandScales[] = { |
+#define VALUE(Name, _) OperandScale::k##Name, |
+ OPERAND_SCALE_LIST(VALUE) |
+#undef VALUE |
+ }; |
+ |
+ for (OperandScale operand_scale : kOperandScales) { |
+#define GENERATE_CODE(Name, ...) \ |
+ InstallBytecodeHandler(interpreter->isolate_, dispatch_table, \ |
+ Bytecode::k##Name, operand_scale); |
+ BYTECODE_LIST(GENERATE_CODE) |
+#undef GENERATE_CODE |
+ } |
+ |
+ // Fill unused entries will the illegal bytecode handler. |
+ size_t illegal_index = Interpreter::GetDispatchTableIndex( |
+ Bytecode::kIllegal, OperandScale::kSingle); |
+ for (size_t index = 0; index < Interpreter::kDispatchTableSize; ++index) { |
+ if (dispatch_table[index] == nullptr) { |
+ dispatch_table[index] = dispatch_table[illegal_index]; |
+ } |
+ } |
+ |
+ // Initialization should have been successful. |
+ DCHECK(interpreter->IsDispatchTableInitialized()); |
+} |
+ |
+// static |
+bool SetupInterpreter::ReuseExistingHandler(Address* dispatch_table, |
+ Bytecode bytecode, |
+ OperandScale operand_scale) { |
+ size_t index = Interpreter::GetDispatchTableIndex(bytecode, operand_scale); |
+ switch (bytecode) { |
+ case Bytecode::kCallProperty: |
+ case Bytecode::kCallProperty0: |
+ case Bytecode::kCallProperty1: |
+ case Bytecode::kCallProperty2: { |
+ const int offset = static_cast<int>(Bytecode::kCallProperty) - |
+ static_cast<int>(Bytecode::kCall); |
+ STATIC_ASSERT(offset == static_cast<int>(Bytecode::kCallProperty0) - |
+ static_cast<int>(Bytecode::kCall0)); |
+ STATIC_ASSERT(offset == static_cast<int>(Bytecode::kCallProperty1) - |
+ static_cast<int>(Bytecode::kCall1)); |
+ STATIC_ASSERT(offset == static_cast<int>(Bytecode::kCallProperty2) - |
+ static_cast<int>(Bytecode::kCall2)); |
+ CHECK_LT(offset, index); |
+ dispatch_table[index] = dispatch_table[index - offset]; |
+ return true; |
+ break; |
+ } |
+ default: |
+ return false; |
+ } |
+} |
+ |
+// static |
+void SetupInterpreter::InstallBytecodeHandler(Isolate* isolate, |
+ Address* dispatch_table, |
+ Bytecode bytecode, |
+ OperandScale operand_scale) { |
+ if (!Bytecodes::BytecodeHasHandler(bytecode, operand_scale)) return; |
+ if (ReuseExistingHandler(dispatch_table, bytecode, operand_scale)) return; |
+ |
+ size_t index = Interpreter::GetDispatchTableIndex(bytecode, operand_scale); |
+ Handle<Code> code = GenerateBytecodeHandler(isolate, bytecode, operand_scale); |
+ dispatch_table[index] = code->entry(); |
+} |
+ |
+} // namespace interpreter |
+} // namespace internal |
+} // namespace v8 |