OLD | NEW |
1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 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/code-generator.h" | 5 #include "src/compiler/code-generator.h" |
6 | 6 |
7 #include "src/compiler/code-generator-impl.h" | 7 #include "src/compiler/code-generator-impl.h" |
8 #include "src/compiler/gap-resolver.h" | 8 #include "src/compiler/gap-resolver.h" |
9 #include "src/compiler/node-matchers.h" | 9 #include "src/compiler/node-matchers.h" |
10 #include "src/compiler/osr.h" | 10 #include "src/compiler/osr.h" |
(...skipping 310 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
321 __ asm_instr(i.MemoryOperand(3), i.InputRegister(2)); \ | 321 __ asm_instr(i.MemoryOperand(3), i.InputRegister(2)); \ |
322 } else { \ | 322 } else { \ |
323 __ asm_instr(i.MemoryOperand(3), i.InputImmediate(2)); \ | 323 __ asm_instr(i.MemoryOperand(3), i.InputImmediate(2)); \ |
324 } \ | 324 } \ |
325 __ bind(&done); \ | 325 __ bind(&done); \ |
326 } while (false) | 326 } while (false) |
327 | 327 |
328 | 328 |
329 void CodeGenerator::AssembleDeconstructActivationRecord(int stack_param_delta) { | 329 void CodeGenerator::AssembleDeconstructActivationRecord(int stack_param_delta) { |
330 CallDescriptor* descriptor = linkage()->GetIncomingDescriptor(); | 330 CallDescriptor* descriptor = linkage()->GetIncomingDescriptor(); |
331 int stack_slots = frame()->GetSpillSlotCount(); | 331 int spill_slots = frame()->GetSpillSlotCount(); |
332 if (descriptor->IsJSFunctionCall() || stack_slots > 0) { | 332 bool has_frame = descriptor->IsJSFunctionCall() || spill_slots > 0; |
333 __ mov(esp, ebp); | 333 if (has_frame) { |
| 334 if (stack_param_delta != 0) { |
| 335 int total_discarded_slots = frame()->GetTotalFrameSlotCount(); |
| 336 // Leave the PC and saved frame pointer on the stack. |
| 337 total_discarded_slots -= |
| 338 StandardFrameConstants::kFixedFrameSizeFromFp / kPointerSize; |
| 339 // Discard only slots that won't be used by new parameters. |
| 340 total_discarded_slots -= stack_param_delta; |
| 341 if (total_discarded_slots > 0) { |
| 342 __ add(esp, Immediate(total_discarded_slots * kPointerSize)); |
| 343 } |
| 344 } else { |
| 345 __ mov(esp, ebp); |
| 346 } |
334 __ pop(ebp); | 347 __ pop(ebp); |
335 } | 348 } |
336 if (stack_param_delta < 0) { | 349 } |
337 int offset = -(stack_param_delta + 1) * kPointerSize; | 350 |
338 __ pop(Operand(esp, offset)); | 351 |
339 if (offset != 0) { | 352 void CodeGenerator::AssemblePrepareTailCall(int stack_param_delta) { |
340 __ add(esp, Immediate(offset)); | 353 if (stack_param_delta > 0) { |
| 354 int total_discarded_slots = frame()->GetTotalFrameSlotCount(); |
| 355 // Leave the PC and saved frame pointer on the stack. |
| 356 total_discarded_slots -= |
| 357 StandardFrameConstants::kFixedFrameSizeFromFp / kPointerSize; |
| 358 // Discard only slots that won't be used by new parameters. |
| 359 total_discarded_slots -= stack_param_delta; |
| 360 if (total_discarded_slots < 0) { |
| 361 __ sub(esp, Immediate(-total_discarded_slots * kPointerSize)); |
341 } | 362 } |
342 } | 363 } |
343 } | 364 } |
344 | 365 |
345 | 366 |
346 // Assembles an instruction after register allocation, producing machine code. | 367 // Assembles an instruction after register allocation, producing machine code. |
347 void CodeGenerator::AssembleArchInstruction(Instruction* instr) { | 368 void CodeGenerator::AssembleArchInstruction(Instruction* instr) { |
348 IA32OperandConverter i(this, instr); | 369 IA32OperandConverter i(this, instr); |
349 | 370 |
350 switch (ArchOpcodeField::decode(instr->opcode())) { | 371 switch (ArchOpcodeField::decode(instr->opcode())) { |
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
401 case kArchLazyBailout: { | 422 case kArchLazyBailout: { |
402 EnsureSpaceForLazyDeopt(); | 423 EnsureSpaceForLazyDeopt(); |
403 RecordCallPosition(instr); | 424 RecordCallPosition(instr); |
404 break; | 425 break; |
405 } | 426 } |
406 case kArchPrepareCallCFunction: { | 427 case kArchPrepareCallCFunction: { |
407 int const num_parameters = MiscField::decode(instr->opcode()); | 428 int const num_parameters = MiscField::decode(instr->opcode()); |
408 __ PrepareCallCFunction(num_parameters, i.TempRegister(0)); | 429 __ PrepareCallCFunction(num_parameters, i.TempRegister(0)); |
409 break; | 430 break; |
410 } | 431 } |
| 432 case kArchPrepareTailCall: |
| 433 AssemblePrepareTailCall(i.InputInt32(instr->InputCount() - 1)); |
| 434 break; |
411 case kArchCallCFunction: { | 435 case kArchCallCFunction: { |
412 int const num_parameters = MiscField::decode(instr->opcode()); | 436 int const num_parameters = MiscField::decode(instr->opcode()); |
413 if (HasImmediateInput(instr, 0)) { | 437 if (HasImmediateInput(instr, 0)) { |
414 ExternalReference ref = i.InputExternalReference(0); | 438 ExternalReference ref = i.InputExternalReference(0); |
415 __ CallCFunction(ref, num_parameters); | 439 __ CallCFunction(ref, num_parameters); |
416 } else { | 440 } else { |
417 Register func = i.InputRegister(0); | 441 Register func = i.InputRegister(0); |
418 __ CallCFunction(func, num_parameters); | 442 __ CallCFunction(func, num_parameters); |
419 } | 443 } |
420 break; | 444 break; |
(...skipping 1194 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1615 int padding_size = last_lazy_deopt_pc_ + space_needed - current_pc; | 1639 int padding_size = last_lazy_deopt_pc_ + space_needed - current_pc; |
1616 __ Nop(padding_size); | 1640 __ Nop(padding_size); |
1617 } | 1641 } |
1618 } | 1642 } |
1619 | 1643 |
1620 #undef __ | 1644 #undef __ |
1621 | 1645 |
1622 } // namespace compiler | 1646 } // namespace compiler |
1623 } // namespace internal | 1647 } // namespace internal |
1624 } // namespace v8 | 1648 } // namespace v8 |
OLD | NEW |