OLD | NEW |
1 // Copyright 2015 the V8 project authors. All rights reserved. | 1 // Copyright 2015 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "src/compiler/interpreter-assembler.h" | 5 #include "src/compiler/interpreter-assembler.h" |
6 | 6 |
7 #include <ostream> | 7 #include <ostream> |
8 | 8 |
9 #include "src/compiler/graph.h" | 9 #include "src/compiler/graph.h" |
10 #include "src/compiler/instruction-selector.h" | 10 #include "src/compiler/instruction-selector.h" |
(...skipping 13 matching lines...) Expand all Loading... |
24 | 24 |
25 | 25 |
26 InterpreterAssembler::InterpreterAssembler(Isolate* isolate, Zone* zone, | 26 InterpreterAssembler::InterpreterAssembler(Isolate* isolate, Zone* zone, |
27 interpreter::Bytecode bytecode) | 27 interpreter::Bytecode bytecode) |
28 : bytecode_(bytecode), | 28 : bytecode_(bytecode), |
29 raw_assembler_(new RawMachineAssembler( | 29 raw_assembler_(new RawMachineAssembler( |
30 isolate, new (zone) Graph(zone), | 30 isolate, new (zone) Graph(zone), |
31 Linkage::GetInterpreterDispatchDescriptor(zone), kMachPtr, | 31 Linkage::GetInterpreterDispatchDescriptor(zone), kMachPtr, |
32 InstructionSelector::SupportedMachineOperatorFlags())), | 32 InstructionSelector::SupportedMachineOperatorFlags())), |
33 end_node_(nullptr), | 33 end_node_(nullptr), |
| 34 outgoing_accumulator_(nullptr), |
34 code_generated_(false) {} | 35 code_generated_(false) {} |
35 | 36 |
36 | 37 |
37 InterpreterAssembler::~InterpreterAssembler() {} | 38 InterpreterAssembler::~InterpreterAssembler() {} |
38 | 39 |
39 | 40 |
40 Handle<Code> InterpreterAssembler::GenerateCode() { | 41 Handle<Code> InterpreterAssembler::GenerateCode() { |
41 DCHECK(!code_generated_); | 42 DCHECK(!code_generated_); |
42 | 43 |
43 End(); | 44 End(); |
44 | 45 |
45 Schedule* schedule = raw_assembler_->Export(); | 46 Schedule* schedule = raw_assembler_->Export(); |
46 // TODO(rmcilroy): use a non-testing code generator. | 47 // TODO(rmcilroy): use a non-testing code generator. |
47 Handle<Code> code = Pipeline::GenerateCodeForTesting( | 48 Handle<Code> code = Pipeline::GenerateCodeForTesting( |
48 isolate(), raw_assembler_->call_descriptor(), graph(), schedule); | 49 isolate(), raw_assembler_->call_descriptor(), graph(), schedule); |
49 | 50 |
50 #ifdef ENABLE_DISASSEMBLER | 51 #ifdef ENABLE_DISASSEMBLER |
51 if (FLAG_trace_ignition_codegen) { | 52 if (FLAG_trace_ignition_codegen) { |
52 OFStream os(stdout); | 53 OFStream os(stdout); |
53 code->Disassemble(interpreter::Bytecodes::ToString(bytecode_), os); | 54 code->Disassemble(interpreter::Bytecodes::ToString(bytecode_), os); |
54 os << std::flush; | 55 os << std::flush; |
55 } | 56 } |
56 #endif | 57 #endif |
57 | 58 |
58 code_generated_ = true; | 59 code_generated_ = true; |
59 return code; | 60 return code; |
60 } | 61 } |
61 | 62 |
62 | 63 |
63 Node* InterpreterAssembler::BytecodeArrayPointer() { | 64 Node* InterpreterAssembler::GetAccumulator() { |
| 65 if (outgoing_accumulator_) { |
| 66 return outgoing_accumulator_; |
| 67 } else { |
| 68 return raw_assembler_->Parameter(Linkage::kInterpreterAccumulatorParameter); |
| 69 } |
| 70 } |
| 71 |
| 72 |
| 73 void InterpreterAssembler::SetAccumulator(Node* value) { |
| 74 outgoing_accumulator_ = value; |
| 75 } |
| 76 |
| 77 |
| 78 Node* InterpreterAssembler::RegisterFileRawPointer() { |
| 79 return raw_assembler_->Parameter(Linkage::kInterpreterRegisterFileParameter); |
| 80 } |
| 81 |
| 82 |
| 83 Node* InterpreterAssembler::BytecodeArrayTaggedPointer() { |
64 return raw_assembler_->Parameter(Linkage::kInterpreterBytecodeArrayParameter); | 84 return raw_assembler_->Parameter(Linkage::kInterpreterBytecodeArrayParameter); |
65 } | 85 } |
66 | 86 |
67 | 87 |
68 Node* InterpreterAssembler::BytecodeOffset() { | 88 Node* InterpreterAssembler::BytecodeOffset() { |
69 return raw_assembler_->Parameter( | 89 return raw_assembler_->Parameter( |
70 Linkage::kInterpreterBytecodeOffsetParameter); | 90 Linkage::kInterpreterBytecodeOffsetParameter); |
71 } | 91 } |
72 | 92 |
73 | 93 |
74 Node* InterpreterAssembler::DispatchTablePointer() { | 94 Node* InterpreterAssembler::DispatchTableRawPointer() { |
75 return raw_assembler_->Parameter(Linkage::kInterpreterDispatchTableParameter); | 95 return raw_assembler_->Parameter(Linkage::kInterpreterDispatchTableParameter); |
76 } | 96 } |
77 | 97 |
78 | 98 |
79 Node* InterpreterAssembler::FramePointer() { | 99 Node* InterpreterAssembler::RegisterFrameOffset(Node* index) { |
80 return raw_assembler_->LoadFramePointer(); | 100 return raw_assembler_->WordShl(index, Int32Constant(kPointerSizeLog2)); |
81 } | 101 } |
82 | 102 |
83 | 103 |
84 Node* InterpreterAssembler::RegisterFrameOffset(int index) { | 104 Node* InterpreterAssembler::LoadRegister(Node* reg_index) { |
85 DCHECK_LE(index, kMaxRegisterIndex); | 105 return raw_assembler_->Load(kMachPtr, RegisterFileRawPointer(), |
86 return Int32Constant(kFirstRegisterOffsetFromFp - | 106 RegisterFrameOffset(reg_index)); |
87 (index << kPointerSizeLog2)); | |
88 } | 107 } |
89 | 108 |
90 | 109 |
91 Node* InterpreterAssembler::RegisterFrameOffset(Node* index) { | 110 Node* InterpreterAssembler::StoreRegister(Node* value, Node* reg_index) { |
92 return raw_assembler_->IntPtrSub( | 111 return raw_assembler_->Store(kMachPtr, RegisterFileRawPointer(), |
93 Int32Constant(kFirstRegisterOffsetFromFp), | 112 RegisterFrameOffset(reg_index), value); |
94 raw_assembler_->WordShl(index, Int32Constant(kPointerSizeLog2))); | |
95 } | 113 } |
96 | 114 |
97 | 115 |
98 Node* InterpreterAssembler::BytecodeOperand(int delta) { | 116 Node* InterpreterAssembler::BytecodeOperand(int delta) { |
99 DCHECK_LT(delta, interpreter::Bytecodes::NumberOfOperands(bytecode_)); | 117 DCHECK_LT(delta, interpreter::Bytecodes::NumberOfOperands(bytecode_)); |
100 return raw_assembler_->Load( | 118 return raw_assembler_->Load( |
101 kMachUint8, BytecodeArrayPointer(), | 119 kMachUint8, BytecodeArrayTaggedPointer(), |
102 raw_assembler_->IntPtrAdd(BytecodeOffset(), Int32Constant(1 + delta))); | 120 raw_assembler_->IntPtrAdd(BytecodeOffset(), Int32Constant(1 + delta))); |
103 } | 121 } |
104 | 122 |
105 | 123 |
106 Node* InterpreterAssembler::LoadRegister(int index) { | 124 Node* InterpreterAssembler::BytecodeOperandSignExtended(int delta) { |
107 return raw_assembler_->Load(kMachPtr, FramePointer(), | 125 DCHECK_LT(delta, interpreter::Bytecodes::NumberOfOperands(bytecode_)); |
108 RegisterFrameOffset(index)); | 126 Node* load = raw_assembler_->Load( |
| 127 kMachInt8, BytecodeArrayTaggedPointer(), |
| 128 raw_assembler_->IntPtrAdd(BytecodeOffset(), Int32Constant(1 + delta))); |
| 129 // Ensure that we sign extend to full pointer size |
| 130 if (kPointerSize == 8) { |
| 131 load = raw_assembler_->ChangeInt32ToInt64(load); |
| 132 } |
| 133 return load; |
109 } | 134 } |
110 | 135 |
111 | 136 |
112 Node* InterpreterAssembler::LoadRegister(Node* index) { | |
113 return raw_assembler_->Load(kMachPtr, FramePointer(), | |
114 RegisterFrameOffset(index)); | |
115 } | |
116 | |
117 | |
118 Node* InterpreterAssembler::StoreRegister(Node* value, int index) { | |
119 return raw_assembler_->Store(kMachPtr, FramePointer(), | |
120 RegisterFrameOffset(index), value); | |
121 } | |
122 | |
123 | |
124 Node* InterpreterAssembler::StoreRegister(Node* value, Node* index) { | |
125 return raw_assembler_->Store(kMachPtr, FramePointer(), | |
126 RegisterFrameOffset(index), value); | |
127 } | |
128 | |
129 | |
130 void InterpreterAssembler::Return() { | 137 void InterpreterAssembler::Return() { |
131 Node* exit_trampoline_code_object = | 138 Node* exit_trampoline_code_object = |
132 HeapConstant(Unique<HeapObject>::CreateImmovable( | 139 HeapConstant(Unique<HeapObject>::CreateImmovable( |
133 isolate()->builtins()->InterpreterExitTrampoline())); | 140 isolate()->builtins()->InterpreterExitTrampoline())); |
134 // If the order of the parameters you need to change the call signature below. | 141 // If the order of the parameters you need to change the call signature below. |
135 STATIC_ASSERT(0 == Linkage::kInterpreterBytecodeOffsetParameter); | 142 STATIC_ASSERT(0 == Linkage::kInterpreterAccumulatorParameter); |
136 STATIC_ASSERT(1 == Linkage::kInterpreterBytecodeArrayParameter); | 143 STATIC_ASSERT(1 == Linkage::kInterpreterRegisterFileParameter); |
137 STATIC_ASSERT(2 == Linkage::kInterpreterDispatchTableParameter); | 144 STATIC_ASSERT(2 == Linkage::kInterpreterBytecodeOffsetParameter); |
138 Node* tail_call = graph()->NewNode( | 145 STATIC_ASSERT(3 == Linkage::kInterpreterBytecodeArrayParameter); |
139 common()->TailCall(call_descriptor()), exit_trampoline_code_object, | 146 STATIC_ASSERT(4 == Linkage::kInterpreterDispatchTableParameter); |
140 BytecodeOffset(), BytecodeArrayPointer(), DispatchTablePointer(), | 147 Node* tail_call = raw_assembler_->TailCallInterpreterDispatch( |
141 graph()->start(), graph()->start()); | 148 call_descriptor(), exit_trampoline_code_object, GetAccumulator(), |
142 schedule()->AddTailCall(raw_assembler_->CurrentBlock(), tail_call); | 149 RegisterFileRawPointer(), BytecodeOffset(), BytecodeArrayTaggedPointer(), |
| 150 DispatchTableRawPointer()); |
143 // This should always be the end node. | 151 // This should always be the end node. |
144 SetEndInput(tail_call); | 152 SetEndInput(tail_call); |
145 } | 153 } |
146 | 154 |
147 | 155 |
148 Node* InterpreterAssembler::Advance(int delta) { | 156 Node* InterpreterAssembler::Advance(int delta) { |
149 return raw_assembler_->IntPtrAdd(BytecodeOffset(), Int32Constant(delta)); | 157 return raw_assembler_->IntPtrAdd(BytecodeOffset(), Int32Constant(delta)); |
150 } | 158 } |
151 | 159 |
152 | 160 |
153 void InterpreterAssembler::Dispatch() { | 161 void InterpreterAssembler::Dispatch() { |
154 Node* new_bytecode_offset = Advance(interpreter::Bytecodes::Size(bytecode_)); | 162 Node* new_bytecode_offset = Advance(interpreter::Bytecodes::Size(bytecode_)); |
155 Node* target_bytecode = raw_assembler_->Load( | 163 Node* target_bytecode = raw_assembler_->Load( |
156 kMachUint8, BytecodeArrayPointer(), new_bytecode_offset); | 164 kMachUint8, BytecodeArrayTaggedPointer(), new_bytecode_offset); |
157 | 165 |
158 // TODO(rmcilroy): Create a code target dispatch table to avoid conversion | 166 // TODO(rmcilroy): Create a code target dispatch table to avoid conversion |
159 // from code object on every dispatch. | 167 // from code object on every dispatch. |
160 Node* target_code_object = raw_assembler_->Load( | 168 Node* target_code_object = raw_assembler_->Load( |
161 kMachPtr, DispatchTablePointer(), | 169 kMachPtr, DispatchTableRawPointer(), |
162 raw_assembler_->Word32Shl(target_bytecode, | 170 raw_assembler_->Word32Shl(target_bytecode, |
163 Int32Constant(kPointerSizeLog2))); | 171 Int32Constant(kPointerSizeLog2))); |
164 | 172 |
165 // If the order of the parameters you need to change the call signature below. | 173 // If the order of the parameters you need to change the call signature below. |
166 STATIC_ASSERT(0 == Linkage::kInterpreterBytecodeOffsetParameter); | 174 STATIC_ASSERT(0 == Linkage::kInterpreterAccumulatorParameter); |
167 STATIC_ASSERT(1 == Linkage::kInterpreterBytecodeArrayParameter); | 175 STATIC_ASSERT(1 == Linkage::kInterpreterRegisterFileParameter); |
168 STATIC_ASSERT(2 == Linkage::kInterpreterDispatchTableParameter); | 176 STATIC_ASSERT(2 == Linkage::kInterpreterBytecodeOffsetParameter); |
169 Node* tail_call = graph()->NewNode( | 177 STATIC_ASSERT(3 == Linkage::kInterpreterBytecodeArrayParameter); |
170 common()->TailCall(call_descriptor()), target_code_object, | 178 STATIC_ASSERT(4 == Linkage::kInterpreterDispatchTableParameter); |
171 new_bytecode_offset, BytecodeArrayPointer(), DispatchTablePointer(), | 179 Node* tail_call = raw_assembler_->TailCallInterpreterDispatch( |
172 graph()->start(), graph()->start()); | 180 call_descriptor(), target_code_object, GetAccumulator(), |
173 schedule()->AddTailCall(raw_assembler_->CurrentBlock(), tail_call); | 181 RegisterFileRawPointer(), new_bytecode_offset, |
| 182 BytecodeArrayTaggedPointer(), DispatchTableRawPointer()); |
174 // This should always be the end node. | 183 // This should always be the end node. |
175 SetEndInput(tail_call); | 184 SetEndInput(tail_call); |
176 } | 185 } |
177 | 186 |
178 | 187 |
179 void InterpreterAssembler::SetEndInput(Node* input) { | 188 void InterpreterAssembler::SetEndInput(Node* input) { |
180 DCHECK(!end_node_); | 189 DCHECK(!end_node_); |
181 end_node_ = input; | 190 end_node_ = input; |
182 } | 191 } |
183 | 192 |
184 | 193 |
185 void InterpreterAssembler::End() { | 194 void InterpreterAssembler::End() { |
186 DCHECK(end_node_); | 195 DCHECK(end_node_); |
187 // TODO(rmcilroy): Support more than 1 end input. | 196 // TODO(rmcilroy): Support more than 1 end input. |
188 Node* end = graph()->NewNode(common()->End(1), end_node_); | 197 Node* end = graph()->NewNode(raw_assembler_->common()->End(1), end_node_); |
189 graph()->SetEnd(end); | 198 graph()->SetEnd(end); |
190 } | 199 } |
191 | 200 |
192 | 201 |
193 // RawMachineAssembler delegate helpers: | 202 // RawMachineAssembler delegate helpers: |
194 Isolate* InterpreterAssembler::isolate() { return raw_assembler_->isolate(); } | 203 Isolate* InterpreterAssembler::isolate() { return raw_assembler_->isolate(); } |
195 | 204 |
196 | 205 |
197 Graph* InterpreterAssembler::graph() { return raw_assembler_->graph(); } | 206 Graph* InterpreterAssembler::graph() { return raw_assembler_->graph(); } |
198 | 207 |
199 | 208 |
200 CallDescriptor* InterpreterAssembler::call_descriptor() const { | 209 CallDescriptor* InterpreterAssembler::call_descriptor() const { |
201 return raw_assembler_->call_descriptor(); | 210 return raw_assembler_->call_descriptor(); |
202 } | 211 } |
203 | 212 |
204 | 213 |
205 Schedule* InterpreterAssembler::schedule() { | 214 Schedule* InterpreterAssembler::schedule() { |
206 return raw_assembler_->schedule(); | 215 return raw_assembler_->schedule(); |
207 } | 216 } |
208 | 217 |
209 | 218 |
210 MachineOperatorBuilder* InterpreterAssembler::machine() { | |
211 return raw_assembler_->machine(); | |
212 } | |
213 | |
214 | |
215 CommonOperatorBuilder* InterpreterAssembler::common() { | |
216 return raw_assembler_->common(); | |
217 } | |
218 | |
219 | |
220 Node* InterpreterAssembler::Int32Constant(int value) { | 219 Node* InterpreterAssembler::Int32Constant(int value) { |
221 return raw_assembler_->Int32Constant(value); | 220 return raw_assembler_->Int32Constant(value); |
222 } | 221 } |
223 | 222 |
224 | 223 |
225 Node* InterpreterAssembler::NumberConstant(double value) { | 224 Node* InterpreterAssembler::NumberConstant(double value) { |
226 return raw_assembler_->NumberConstant(value); | 225 return raw_assembler_->NumberConstant(value); |
227 } | 226 } |
228 | 227 |
229 | 228 |
230 Node* InterpreterAssembler::HeapConstant(Unique<HeapObject> object) { | 229 Node* InterpreterAssembler::HeapConstant(Unique<HeapObject> object) { |
231 return raw_assembler_->HeapConstant(object); | 230 return raw_assembler_->HeapConstant(object); |
232 } | 231 } |
233 | 232 |
234 | |
235 } // namespace interpreter | 233 } // namespace interpreter |
236 } // namespace internal | 234 } // namespace internal |
237 } // namespace v8 | 235 } // namespace v8 |
OLD | NEW |