1 // Copyright 2014 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "src/compiler/code-generator.h"
6
7 #include "src/arm/macro-assembler-arm.h"
8 #include "src/compiler/code-generator-impl.h"
9 #include "src/compiler/gap-resolver.h"
10 #include "src/compiler/node-matchers.h"
11 #include "src/compiler/node-properties-inl.h"
12 #include "src/scopes.h"
13
14 namespace v8 {
15 namespace internal {
16 namespace compiler {
17
18 #define __ masm()->
19
20
21 #define kScratchReg r9
22
23
24 // Adds Arm-specific methods to convert InstructionOperands.
25 class ArmOperandConverter : public InstructionOperandConverter {
26 public:
ArmOperandConverter(CodeGenerator * gen,Instruction * instr)27 ArmOperandConverter(CodeGenerator* gen, Instruction* instr)
28 : InstructionOperandConverter(gen, instr) {}
29
OutputSBit() const30 SBit OutputSBit() const {
31 switch (instr_->flags_mode()) {
32 case kFlags_branch:
33 case kFlags_set:
34 return SetCC;
35 case kFlags_none:
36 return LeaveCC;
37 }
38 UNREACHABLE();
39 return LeaveCC;
40 }
41
InputImmediate(int index)42 Operand InputImmediate(int index) {
43 Constant constant = ToConstant(instr_->InputAt(index));
44 switch (constant.type()) {
45 case Constant::kInt32:
46 return Operand(constant.ToInt32());
47 case Constant::kFloat64:
48 return Operand(
49 isolate()->factory()->NewNumber(constant.ToFloat64(), TENURED));
50 case Constant::kInt64:
51 case Constant::kExternalReference:
52 case Constant::kHeapObject:
53 break;
54 }
55 UNREACHABLE();
56 return Operand::Zero();
57 }
58
InputOperand2(int first_index)59 Operand InputOperand2(int first_index) {
60 const int index = first_index;
61 switch (AddressingModeField::decode(instr_->opcode())) {
62 case kMode_None:
63 case kMode_Offset_RI:
64 case kMode_Offset_RR:
65 break;
66 case kMode_Operand2_I:
67 return InputImmediate(index + 0);
68 case kMode_Operand2_R:
69 return Operand(InputRegister(index + 0));
70 case kMode_Operand2_R_ASR_I:
71 return Operand(InputRegister(index + 0), ASR, InputInt5(index + 1));
72 case kMode_Operand2_R_ASR_R:
73 return Operand(InputRegister(index + 0), ASR, InputRegister(index + 1));
74 case kMode_Operand2_R_LSL_I:
75 return Operand(InputRegister(index + 0), LSL, InputInt5(index + 1));
76 case kMode_Operand2_R_LSL_R:
77 return Operand(InputRegister(index + 0), LSL, InputRegister(index + 1));
78 case kMode_Operand2_R_LSR_I:
79 return Operand(InputRegister(index + 0), LSR, InputInt5(index + 1));
80 case kMode_Operand2_R_LSR_R:
81 return Operand(InputRegister(index + 0), LSR, InputRegister(index + 1));
82 case kMode_Operand2_R_ROR_I:
83 return Operand(InputRegister(index + 0), ROR, InputInt5(index + 1));
84 case kMode_Operand2_R_ROR_R:
85 return Operand(InputRegister(index + 0), ROR, InputRegister(index + 1));
86 }
87 UNREACHABLE();
88 return Operand::Zero();
89 }
90
InputOffset(int * first_index)91 MemOperand InputOffset(int* first_index) {
92 const int index = *first_index;
93 switch (AddressingModeField::decode(instr_->opcode())) {
94 case kMode_None:
95 case kMode_Operand2_I:
96 case kMode_Operand2_R:
97 case kMode_Operand2_R_ASR_I:
98 case kMode_Operand2_R_ASR_R:
99 case kMode_Operand2_R_LSL_I:
100 case kMode_Operand2_R_LSL_R:
101 case kMode_Operand2_R_LSR_I:
102 case kMode_Operand2_R_LSR_R:
103 case kMode_Operand2_R_ROR_I:
104 case kMode_Operand2_R_ROR_R:
105 break;
106 case kMode_Offset_RI:
107 *first_index += 2;
108 return MemOperand(InputRegister(index + 0), InputInt32(index + 1));
109 case kMode_Offset_RR:
110 *first_index += 2;
111 return MemOperand(InputRegister(index + 0), InputRegister(index + 1));
112 }
113 UNREACHABLE();
114 return MemOperand(r0);
115 }
116
InputOffset()117 MemOperand InputOffset() {
118 int index = 0;
119 return InputOffset(&index);
120 }
121
ToMemOperand(InstructionOperand * op) const122 MemOperand ToMemOperand(InstructionOperand* op) const {
123 DCHECK(op != NULL);
124 DCHECK(!op->IsRegister());
125 DCHECK(!op->IsDoubleRegister());
126 DCHECK(op->IsStackSlot() || op->IsDoubleStackSlot());
127 // The linkage computes where all spill slots are located.
128 FrameOffset offset = linkage()->GetFrameOffset(op->index(), frame(), 0);
129 return MemOperand(offset.from_stack_pointer() ? sp : fp, offset.offset());
130 }
131 };
132
133
134 // Assembles an instruction after register allocation, producing machine code.
AssembleArchInstruction(Instruction * instr)135 void CodeGenerator::AssembleArchInstruction(Instruction* instr) {
136 ArmOperandConverter i(this, instr);
137
138 switch (ArchOpcodeField::decode(instr->opcode())) {
139 case kArchCallCodeObject: {
140 EnsureSpaceForLazyDeopt();
141 if (instr->InputAt(0)->IsImmediate()) {
142 __ Call(Handle<Code>::cast(i.InputHeapObject(0)),
143 RelocInfo::CODE_TARGET);
144 } else {
145 __ add(ip, i.InputRegister(0),
146 Operand(Code::kHeaderSize - kHeapObjectTag));
147 __ Call(ip);
148 }
149 AddSafepointAndDeopt(instr);
150 DCHECK_EQ(LeaveCC, i.OutputSBit());
151 break;
152 }
153 case kArchCallJSFunction: {
154 EnsureSpaceForLazyDeopt();
155 Register func = i.InputRegister(0);
156 if (FLAG_debug_code) {
157 // Check the function's context matches the context argument.
158 __ ldr(kScratchReg, FieldMemOperand(func, JSFunction::kContextOffset));
159 __ cmp(cp, kScratchReg);
160 __ Assert(eq, kWrongFunctionContext);
161 }
162 __ ldr(ip, FieldMemOperand(func, JSFunction::kCodeEntryOffset));
163 __ Call(ip);
164 AddSafepointAndDeopt(instr);
165 DCHECK_EQ(LeaveCC, i.OutputSBit());
166 break;
167 }
168 case kArchJmp:
169 __ b(code_->GetLabel(i.InputBlock(0)));
170 DCHECK_EQ(LeaveCC, i.OutputSBit());
171 break;
172 case kArchNop:
173 // don't emit code for nops.
174 DCHECK_EQ(LeaveCC, i.OutputSBit());
175 break;
176 case kArchRet:
177 AssembleReturn();
178 DCHECK_EQ(LeaveCC, i.OutputSBit());
179 break;
180 case kArchTruncateDoubleToI:
181 __ TruncateDoubleToI(i.OutputRegister(), i.InputDoubleRegister(0));
182 DCHECK_EQ(LeaveCC, i.OutputSBit());
183 break;
184 case kArmAdd:
185 __ add(i.OutputRegister(), i.InputRegister(0), i.InputOperand2(1),
186 i.OutputSBit());
187 break;
188 case kArmAnd:
189 __ and_(i.OutputRegister(), i.InputRegister(0), i.InputOperand2(1),
190 i.OutputSBit());
191 break;
192 case kArmBic:
193 __ bic(i.OutputRegister(), i.InputRegister(0), i.InputOperand2(1),
194 i.OutputSBit());
195 break;
196 case kArmMul:
197 __ mul(i.OutputRegister(), i.InputRegister(0), i.InputRegister(1),
198 i.OutputSBit());
199 break;
200 case kArmMla:
201 __ mla(i.OutputRegister(), i.InputRegister(0), i.InputRegister(1),
202 i.InputRegister(2), i.OutputSBit());
203 break;
204 case kArmMls: {
205 CpuFeatureScope scope(masm(), MLS);
206 __ mls(i.OutputRegister(), i.InputRegister(0), i.InputRegister(1),
207 i.InputRegister(2));
208 DCHECK_EQ(LeaveCC, i.OutputSBit());
209 break;
210 }
211 case kArmSdiv: {
212 CpuFeatureScope scope(masm(), SUDIV);
213 __ sdiv(i.OutputRegister(), i.InputRegister(0), i.InputRegister(1));
214 DCHECK_EQ(LeaveCC, i.OutputSBit());
215 break;
216 }
217 case kArmUdiv: {
218 CpuFeatureScope scope(masm(), SUDIV);
219 __ udiv(i.OutputRegister(), i.InputRegister(0), i.InputRegister(1));
220 DCHECK_EQ(LeaveCC, i.OutputSBit());
221 break;
222 }
223 case kArmMov:
224 __ Move(i.OutputRegister(), i.InputOperand2(0), i.OutputSBit());
225 break;
226 case kArmMvn:
227 __ mvn(i.OutputRegister(), i.InputOperand2(0), i.OutputSBit());
228 break;
229 case kArmOrr:
230 __ orr(i.OutputRegister(), i.InputRegister(0), i.InputOperand2(1),
231 i.OutputSBit());
232 break;
233 case kArmEor:
234 __ eor(i.OutputRegister(), i.InputRegister(0), i.InputOperand2(1),
235 i.OutputSBit());
236 break;
237 case kArmSub:
238 __ sub(i.OutputRegister(), i.InputRegister(0), i.InputOperand2(1),
239 i.OutputSBit());
240 break;
241 case kArmRsb:
242 __ rsb(i.OutputRegister(), i.InputRegister(0), i.InputOperand2(1),
243 i.OutputSBit());
244 break;
245 case kArmBfc: {
246 CpuFeatureScope scope(masm(), ARMv7);
247 __ bfc(i.OutputRegister(), i.InputInt8(1), i.InputInt8(2));
248 DCHECK_EQ(LeaveCC, i.OutputSBit());
249 break;
250 }
251 case kArmUbfx: {
252 CpuFeatureScope scope(masm(), ARMv7);
253 __ ubfx(i.OutputRegister(), i.InputRegister(0), i.InputInt8(1),
254 i.InputInt8(2));
255 DCHECK_EQ(LeaveCC, i.OutputSBit());
256 break;
257 }
258 case kArmCmp:
259 __ cmp(i.InputRegister(0), i.InputOperand2(1));
260 DCHECK_EQ(SetCC, i.OutputSBit());
261 break;
262 case kArmCmn:
263 __ cmn(i.InputRegister(0), i.InputOperand2(1));
264 DCHECK_EQ(SetCC, i.OutputSBit());
265 break;
266 case kArmTst:
267 __ tst(i.InputRegister(0), i.InputOperand2(1));
268 DCHECK_EQ(SetCC, i.OutputSBit());
269 break;
270 case kArmTeq:
271 __ teq(i.InputRegister(0), i.InputOperand2(1));
272 DCHECK_EQ(SetCC, i.OutputSBit());
273 break;
274 case kArmVcmpF64:
275 __ VFPCompareAndSetFlags(i.InputDoubleRegister(0),
276 i.InputDoubleRegister(1));
277 DCHECK_EQ(SetCC, i.OutputSBit());
278 break;
279 case kArmVaddF64:
280 __ vadd(i.OutputDoubleRegister(), i.InputDoubleRegister(0),
281 i.InputDoubleRegister(1));
282 DCHECK_EQ(LeaveCC, i.OutputSBit());
283 break;
284 case kArmVsubF64:
285 __ vsub(i.OutputDoubleRegister(), i.InputDoubleRegister(0),
286 i.InputDoubleRegister(1));
287 DCHECK_EQ(LeaveCC, i.OutputSBit());
288 break;
289 case kArmVmulF64:
290 __ vmul(i.OutputDoubleRegister(), i.InputDoubleRegister(0),
291 i.InputDoubleRegister(1));
292 DCHECK_EQ(LeaveCC, i.OutputSBit());
293 break;
294 case kArmVmlaF64:
295 __ vmla(i.OutputDoubleRegister(), i.InputDoubleRegister(1),
296 i.InputDoubleRegister(2));
297 DCHECK_EQ(LeaveCC, i.OutputSBit());
298 break;
299 case kArmVmlsF64:
300 __ vmls(i.OutputDoubleRegister(), i.InputDoubleRegister(1),
301 i.InputDoubleRegister(2));
302 DCHECK_EQ(LeaveCC, i.OutputSBit());
303 break;
304 case kArmVdivF64:
305 __ vdiv(i.OutputDoubleRegister(), i.InputDoubleRegister(0),
306 i.InputDoubleRegister(1));
307 DCHECK_EQ(LeaveCC, i.OutputSBit());
308 break;
309 case kArmVmodF64: {
310 // TODO(bmeurer): We should really get rid of this special instruction,
311 // and generate a CallAddress instruction instead.
312 FrameScope scope(masm(), StackFrame::MANUAL);
313 __ PrepareCallCFunction(0, 2, kScratchReg);
314 __ MovToFloatParameters(i.InputDoubleRegister(0),
315 i.InputDoubleRegister(1));
316 __ CallCFunction(ExternalReference::mod_two_doubles_operation(isolate()),
317 0, 2);
318 // Move the result in the double result register.
319 __ MovFromFloatResult(i.OutputDoubleRegister());
320 DCHECK_EQ(LeaveCC, i.OutputSBit());
321 break;
322 }
323 case kArmVnegF64:
324 __ vneg(i.OutputDoubleRegister(), i.InputDoubleRegister(0));
325 break;
326 case kArmVsqrtF64:
327 __ vsqrt(i.OutputDoubleRegister(), i.InputDoubleRegister(0));
328 break;
329 case kArmVcvtF64S32: {
330 SwVfpRegister scratch = kScratchDoubleReg.low();
331 __ vmov(scratch, i.InputRegister(0));
332 __ vcvt_f64_s32(i.OutputDoubleRegister(), scratch);
333 DCHECK_EQ(LeaveCC, i.OutputSBit());
334 break;
335 }
336 case kArmVcvtF64U32: {
337 SwVfpRegister scratch = kScratchDoubleReg.low();
338 __ vmov(scratch, i.InputRegister(0));
339 __ vcvt_f64_u32(i.OutputDoubleRegister(), scratch);
340 DCHECK_EQ(LeaveCC, i.OutputSBit());
341 break;
342 }
343 case kArmVcvtS32F64: {
344 SwVfpRegister scratch = kScratchDoubleReg.low();
345 __ vcvt_s32_f64(scratch, i.InputDoubleRegister(0));
346 __ vmov(i.OutputRegister(), scratch);
347 DCHECK_EQ(LeaveCC, i.OutputSBit());
348 break;
349 }
350 case kArmVcvtU32F64: {
351 SwVfpRegister scratch = kScratchDoubleReg.low();
352 __ vcvt_u32_f64(scratch, i.InputDoubleRegister(0));
353 __ vmov(i.OutputRegister(), scratch);
354 DCHECK_EQ(LeaveCC, i.OutputSBit());
355 break;
356 }
357 case kArmLdrb:
358 __ ldrb(i.OutputRegister(), i.InputOffset());
359 DCHECK_EQ(LeaveCC, i.OutputSBit());
360 break;
361 case kArmLdrsb:
362 __ ldrsb(i.OutputRegister(), i.InputOffset());
363 DCHECK_EQ(LeaveCC, i.OutputSBit());
364 break;
365 case kArmStrb: {
366 int index = 0;
367 MemOperand operand = i.InputOffset(&index);
368 __ strb(i.InputRegister(index), operand);
369 DCHECK_EQ(LeaveCC, i.OutputSBit());
370 break;
371 }
372 case kArmLdrh:
373 __ ldrh(i.OutputRegister(), i.InputOffset());
374 break;
375 case kArmLdrsh:
376 __ ldrsh(i.OutputRegister(), i.InputOffset());
377 break;
378 case kArmStrh: {
379 int index = 0;
380 MemOperand operand = i.InputOffset(&index);
381 __ strh(i.InputRegister(index), operand);
382 DCHECK_EQ(LeaveCC, i.OutputSBit());
383 break;
384 }
385 case kArmLdr:
386 __ ldr(i.OutputRegister(), i.InputOffset());
387 break;
388 case kArmStr: {
389 int index = 0;
390 MemOperand operand = i.InputOffset(&index);
391 __ str(i.InputRegister(index), operand);
392 DCHECK_EQ(LeaveCC, i.OutputSBit());
393 break;
394 }
395 case kArmVldr32: {
396 SwVfpRegister scratch = kScratchDoubleReg.low();
397 __ vldr(scratch, i.InputOffset());
398 __ vcvt_f64_f32(i.OutputDoubleRegister(), scratch);
399 DCHECK_EQ(LeaveCC, i.OutputSBit());
400 break;
401 }
402 case kArmVstr32: {
403 int index = 0;
404 SwVfpRegister scratch = kScratchDoubleReg.low();
405 MemOperand operand = i.InputOffset(&index);
406 __ vcvt_f32_f64(scratch, i.InputDoubleRegister(index));
407 __ vstr(scratch, operand);
408 DCHECK_EQ(LeaveCC, i.OutputSBit());
409 break;
410 }
411 case kArmVldr64:
412 __ vldr(i.OutputDoubleRegister(), i.InputOffset());
413 DCHECK_EQ(LeaveCC, i.OutputSBit());
414 break;
415 case kArmVstr64: {
416 int index = 0;
417 MemOperand operand = i.InputOffset(&index);
418 __ vstr(i.InputDoubleRegister(index), operand);
419 DCHECK_EQ(LeaveCC, i.OutputSBit());
420 break;
421 }
422 case kArmPush:
423 __ Push(i.InputRegister(0));
424 DCHECK_EQ(LeaveCC, i.OutputSBit());
425 break;
426 case kArmStoreWriteBarrier: {
427 Register object = i.InputRegister(0);
428 Register index = i.InputRegister(1);
429 Register value = i.InputRegister(2);
430 __ add(index, object, index);
431 __ str(value, MemOperand(index));
432 SaveFPRegsMode mode =
433 frame()->DidAllocateDoubleRegisters() ? kSaveFPRegs : kDontSaveFPRegs;
434 LinkRegisterStatus lr_status = kLRHasNotBeenSaved;
435 __ RecordWrite(object, index, value, lr_status, mode);
436 DCHECK_EQ(LeaveCC, i.OutputSBit());
437 break;
438 }
439 }
440 }
441
442
443 // Assembles branches after an instruction.
AssembleArchBranch(Instruction * instr,FlagsCondition condition)444 void CodeGenerator::AssembleArchBranch(Instruction* instr,
445 FlagsCondition condition) {
446 ArmOperandConverter i(this, instr);
447 Label done;
448
449 // Emit a branch. The true and false targets are always the last two inputs
450 // to the instruction.
451 BasicBlock* tblock = i.InputBlock(instr->InputCount() - 2);
452 BasicBlock* fblock = i.InputBlock(instr->InputCount() - 1);
453 bool fallthru = IsNextInAssemblyOrder(fblock);
454 Label* tlabel = code()->GetLabel(tblock);
455 Label* flabel = fallthru ? &done : code()->GetLabel(fblock);
456 switch (condition) {
457 case kUnorderedEqual:
458 __ b(vs, flabel);
459 // Fall through.
460 case kEqual:
461 __ b(eq, tlabel);
462 break;
463 case kUnorderedNotEqual:
464 __ b(vs, tlabel);
465 // Fall through.
466 case kNotEqual:
467 __ b(ne, tlabel);
468 break;
469 case kSignedLessThan:
470 __ b(lt, tlabel);
471 break;
472 case kSignedGreaterThanOrEqual:
473 __ b(ge, tlabel);
474 break;
475 case kSignedLessThanOrEqual:
476 __ b(le, tlabel);
477 break;
478 case kSignedGreaterThan:
479 __ b(gt, tlabel);
480 break;
481 case kUnorderedLessThan:
482 __ b(vs, flabel);
483 // Fall through.
484 case kUnsignedLessThan:
485 __ b(lo, tlabel);
486 break;
487 case kUnorderedGreaterThanOrEqual:
488 __ b(vs, tlabel);
489 // Fall through.
490 case kUnsignedGreaterThanOrEqual:
491 __ b(hs, tlabel);
492 break;
493 case kUnorderedLessThanOrEqual:
494 __ b(vs, flabel);
495 // Fall through.
496 case kUnsignedLessThanOrEqual:
497 __ b(ls, tlabel);
498 break;
499 case kUnorderedGreaterThan:
500 __ b(vs, tlabel);
501 // Fall through.
502 case kUnsignedGreaterThan:
503 __ b(hi, tlabel);
504 break;
505 case kOverflow:
506 __ b(vs, tlabel);
507 break;
508 case kNotOverflow:
509 __ b(vc, tlabel);
510 break;
511 }
512 if (!fallthru) __ b(flabel); // no fallthru to flabel.
513 __ bind(&done);
514 }
515
516
517 // Assembles boolean materializations after an instruction.
AssembleArchBoolean(Instruction * instr,FlagsCondition condition)518 void CodeGenerator::AssembleArchBoolean(Instruction* instr,
519 FlagsCondition condition) {
520 ArmOperandConverter i(this, instr);
521 Label done;
522
523 // Materialize a full 32-bit 1 or 0 value. The result register is always the
524 // last output of the instruction.
525 Label check;
526 DCHECK_NE(0, instr->OutputCount());
527 Register reg = i.OutputRegister(instr->OutputCount() - 1);
528 Condition cc = kNoCondition;
529 switch (condition) {
530 case kUnorderedEqual:
531 __ b(vc, &check);
532 __ mov(reg, Operand(0));
533 __ b(&done);
534 // Fall through.
535 case kEqual:
536 cc = eq;
537 break;
538 case kUnorderedNotEqual:
539 __ b(vc, &check);
540 __ mov(reg, Operand(1));
541 __ b(&done);
542 // Fall through.
543 case kNotEqual:
544 cc = ne;
545 break;
546 case kSignedLessThan:
547 cc = lt;
548 break;
549 case kSignedGreaterThanOrEqual:
550 cc = ge;
551 break;
552 case kSignedLessThanOrEqual:
553 cc = le;
554 break;
555 case kSignedGreaterThan:
556 cc = gt;
557 break;
558 case kUnorderedLessThan:
559 __ b(vc, &check);
560 __ mov(reg, Operand(0));
561 __ b(&done);
562 // Fall through.
563 case kUnsignedLessThan:
564 cc = lo;
565 break;
566 case kUnorderedGreaterThanOrEqual:
567 __ b(vc, &check);
568 __ mov(reg, Operand(1));
569 __ b(&done);
570 // Fall through.
571 case kUnsignedGreaterThanOrEqual:
572 cc = hs;
573 break;
574 case kUnorderedLessThanOrEqual:
575 __ b(vc, &check);
576 __ mov(reg, Operand(0));
577 __ b(&done);
578 // Fall through.
579 case kUnsignedLessThanOrEqual:
580 cc = ls;
581 break;
582 case kUnorderedGreaterThan:
583 __ b(vc, &check);
584 __ mov(reg, Operand(1));
585 __ b(&done);
586 // Fall through.
587 case kUnsignedGreaterThan:
588 cc = hi;
589 break;
590 case kOverflow:
591 cc = vs;
592 break;
593 case kNotOverflow:
594 cc = vc;
595 break;
596 }
597 __ bind(&check);
598 __ mov(reg, Operand(0));
599 __ mov(reg, Operand(1), LeaveCC, cc);
600 __ bind(&done);
601 }
602
603
AssembleDeoptimizerCall(int deoptimization_id)604 void CodeGenerator::AssembleDeoptimizerCall(int deoptimization_id) {
605 Address deopt_entry = Deoptimizer::GetDeoptimizationEntry(
606 isolate(), deoptimization_id, Deoptimizer::LAZY);
607 __ Call(deopt_entry, RelocInfo::RUNTIME_ENTRY);
608 }
609
610
AssemblePrologue()611 void CodeGenerator::AssemblePrologue() {
612 CallDescriptor* descriptor = linkage()->GetIncomingDescriptor();
613 if (descriptor->kind() == CallDescriptor::kCallAddress) {
614 bool saved_pp;
615 if (FLAG_enable_ool_constant_pool) {
616 __ Push(lr, fp, pp);
617 // Adjust FP to point to saved FP.
618 __ sub(fp, sp, Operand(StandardFrameConstants::kConstantPoolOffset));
619 saved_pp = true;
620 } else {
621 __ Push(lr, fp);
622 __ mov(fp, sp);
623 saved_pp = false;
624 }
625 const RegList saves = descriptor->CalleeSavedRegisters();
626 if (saves != 0 || saved_pp) {
627 // Save callee-saved registers.
628 int register_save_area_size = saved_pp ? kPointerSize : 0;
629 for (int i = Register::kNumRegisters - 1; i >= 0; i--) {
630 if (!((1 << i) & saves)) continue;
631 register_save_area_size += kPointerSize;
632 }
633 frame()->SetRegisterSaveAreaSize(register_save_area_size);
634 __ stm(db_w, sp, saves);
635 }
636 } else if (descriptor->IsJSFunctionCall()) {
637 CompilationInfo* info = linkage()->info();
638 __ Prologue(info->IsCodePreAgingActive());
639 frame()->SetRegisterSaveAreaSize(
640 StandardFrameConstants::kFixedFrameSizeFromFp);
641
642 // Sloppy mode functions and builtins need to replace the receiver with the
643 // global proxy when called as functions (without an explicit receiver
644 // object).
645 // TODO(mstarzinger/verwaest): Should this be moved back into the CallIC?
646 if (info->strict_mode() == SLOPPY && !info->is_native()) {
647 Label ok;
648 // +2 for return address and saved frame pointer.
649 int receiver_slot = info->scope()->num_parameters() + 2;
650 __ ldr(r2, MemOperand(fp, receiver_slot * kPointerSize));
651 __ CompareRoot(r2, Heap::kUndefinedValueRootIndex);
652 __ b(ne, &ok);
653 __ ldr(r2, GlobalObjectOperand());
654 __ ldr(r2, FieldMemOperand(r2, GlobalObject::kGlobalProxyOffset));
655 __ str(r2, MemOperand(fp, receiver_slot * kPointerSize));
656 __ bind(&ok);
657 }
658
659 } else {
660 __ StubPrologue();
661 frame()->SetRegisterSaveAreaSize(
662 StandardFrameConstants::kFixedFrameSizeFromFp);
663 }
664 int stack_slots = frame()->GetSpillSlotCount();
665 if (stack_slots > 0) {
666 __ sub(sp, sp, Operand(stack_slots * kPointerSize));
667 }
668 }
669
670
AssembleReturn()671 void CodeGenerator::AssembleReturn() {
672 CallDescriptor* descriptor = linkage()->GetIncomingDescriptor();
673 if (descriptor->kind() == CallDescriptor::kCallAddress) {
674 if (frame()->GetRegisterSaveAreaSize() > 0) {
675 // Remove this frame's spill slots first.
676 int stack_slots = frame()->GetSpillSlotCount();
677 if (stack_slots > 0) {
678 __ add(sp, sp, Operand(stack_slots * kPointerSize));
679 }
680 // Restore registers.
681 const RegList saves = descriptor->CalleeSavedRegisters();
682 if (saves != 0) {
683 __ ldm(ia_w, sp, saves);
684 }
685 }
686 __ LeaveFrame(StackFrame::MANUAL);
687 __ Ret();
688 } else {
689 __ LeaveFrame(StackFrame::MANUAL);
690 int pop_count = descriptor->IsJSFunctionCall()
691 ? static_cast<int>(descriptor->JSParameterCount())
692 : 0;
693 __ Drop(pop_count);
694 __ Ret();
695 }
696 }
697
698
AssembleMove(InstructionOperand * source,InstructionOperand * destination)699 void CodeGenerator::AssembleMove(InstructionOperand* source,
700 InstructionOperand* destination) {
701 ArmOperandConverter g(this, NULL);
702 // Dispatch on the source and destination operand kinds. Not all
703 // combinations are possible.
704 if (source->IsRegister()) {
705 DCHECK(destination->IsRegister() || destination->IsStackSlot());
706 Register src = g.ToRegister(source);
707 if (destination->IsRegister()) {
708 __ mov(g.ToRegister(destination), src);
709 } else {
710 __ str(src, g.ToMemOperand(destination));
711 }
712 } else if (source->IsStackSlot()) {
713 DCHECK(destination->IsRegister() || destination->IsStackSlot());
714 MemOperand src = g.ToMemOperand(source);
715 if (destination->IsRegister()) {
716 __ ldr(g.ToRegister(destination), src);
717 } else {
718 Register temp = kScratchReg;
719 __ ldr(temp, src);
720 __ str(temp, g.ToMemOperand(destination));
721 }
722 } else if (source->IsConstant()) {
723 if (destination->IsRegister() || destination->IsStackSlot()) {
724 Register dst =
725 destination->IsRegister() ? g.ToRegister(destination) : kScratchReg;
726 Constant src = g.ToConstant(source);
727 switch (src.type()) {
728 case Constant::kInt32:
729 __ mov(dst, Operand(src.ToInt32()));
730 break;
731 case Constant::kInt64:
732 UNREACHABLE();
733 break;
734 case Constant::kFloat64:
735 __ Move(dst,
736 isolate()->factory()->NewNumber(src.ToFloat64(), TENURED));
737 break;
738 case Constant::kExternalReference:
739 __ mov(dst, Operand(src.ToExternalReference()));
740 break;
741 case Constant::kHeapObject:
742 __ Move(dst, src.ToHeapObject());
743 break;
744 }
745 if (destination->IsStackSlot()) __ str(dst, g.ToMemOperand(destination));
746 } else if (destination->IsDoubleRegister()) {
747 DwVfpRegister result = g.ToDoubleRegister(destination);
748 __ vmov(result, g.ToDouble(source));
749 } else {
750 DCHECK(destination->IsDoubleStackSlot());
751 DwVfpRegister temp = kScratchDoubleReg;
752 __ vmov(temp, g.ToDouble(source));
753 __ vstr(temp, g.ToMemOperand(destination));
754 }
755 } else if (source->IsDoubleRegister()) {
756 DwVfpRegister src = g.ToDoubleRegister(source);
757 if (destination->IsDoubleRegister()) {
758 DwVfpRegister dst = g.ToDoubleRegister(destination);
759 __ Move(dst, src);
760 } else {
761 DCHECK(destination->IsDoubleStackSlot());
762 __ vstr(src, g.ToMemOperand(destination));
763 }
764 } else if (source->IsDoubleStackSlot()) {
765 DCHECK(destination->IsDoubleRegister() || destination->IsDoubleStackSlot());
766 MemOperand src = g.ToMemOperand(source);
767 if (destination->IsDoubleRegister()) {
768 __ vldr(g.ToDoubleRegister(destination), src);
769 } else {
770 DwVfpRegister temp = kScratchDoubleReg;
771 __ vldr(temp, src);
772 __ vstr(temp, g.ToMemOperand(destination));
773 }
774 } else {
775 UNREACHABLE();
776 }
777 }
778
779
AssembleSwap(InstructionOperand * source,InstructionOperand * destination)780 void CodeGenerator::AssembleSwap(InstructionOperand* source,
781 InstructionOperand* destination) {
782 ArmOperandConverter g(this, NULL);
783 // Dispatch on the source and destination operand kinds. Not all
784 // combinations are possible.
785 if (source->IsRegister()) {
786 // Register-register.
787 Register temp = kScratchReg;
788 Register src = g.ToRegister(source);
789 if (destination->IsRegister()) {
790 Register dst = g.ToRegister(destination);
791 __ Move(temp, src);
792 __ Move(src, dst);
793 __ Move(dst, temp);
794 } else {
795 DCHECK(destination->IsStackSlot());
796 MemOperand dst = g.ToMemOperand(destination);
797 __ mov(temp, src);
798 __ ldr(src, dst);
799 __ str(temp, dst);
800 }
801 } else if (source->IsStackSlot()) {
802 DCHECK(destination->IsStackSlot());
803 Register temp_0 = kScratchReg;
804 SwVfpRegister temp_1 = kScratchDoubleReg.low();
805 MemOperand src = g.ToMemOperand(source);
806 MemOperand dst = g.ToMemOperand(destination);
807 __ ldr(temp_0, src);
808 __ vldr(temp_1, dst);
809 __ str(temp_0, dst);
810 __ vstr(temp_1, src);
811 } else if (source->IsDoubleRegister()) {
812 DwVfpRegister temp = kScratchDoubleReg;
813 DwVfpRegister src = g.ToDoubleRegister(source);
814 if (destination->IsDoubleRegister()) {
815 DwVfpRegister dst = g.ToDoubleRegister(destination);
816 __ Move(temp, src);
817 __ Move(src, dst);
818 __ Move(dst, temp);
819 } else {
820 DCHECK(destination->IsDoubleStackSlot());
821 MemOperand dst = g.ToMemOperand(destination);
822 __ Move(temp, src);
823 __ vldr(src, dst);
824 __ vstr(temp, dst);
825 }
826 } else if (source->IsDoubleStackSlot()) {
827 DCHECK(destination->IsDoubleStackSlot());
828 Register temp_0 = kScratchReg;
829 DwVfpRegister temp_1 = kScratchDoubleReg;
830 MemOperand src0 = g.ToMemOperand(source);
831 MemOperand src1(src0.rn(), src0.offset() + kPointerSize);
832 MemOperand dst0 = g.ToMemOperand(destination);
833 MemOperand dst1(dst0.rn(), dst0.offset() + kPointerSize);
834 __ vldr(temp_1, dst0); // Save destination in temp_1.
835 __ ldr(temp_0, src0); // Then use temp_0 to copy source to destination.
836 __ str(temp_0, dst0);
837 __ ldr(temp_0, src1);
838 __ str(temp_0, dst1);
839 __ vstr(temp_1, src0);
840 } else {
841 // No other combinations are possible.
842 UNREACHABLE();
843 }
844 }
845
846
AddNopForSmiCodeInlining()847 void CodeGenerator::AddNopForSmiCodeInlining() {
848 // On 32-bit ARM we do not insert nops for inlined Smi code.
849 }
850
851
EnsureSpaceForLazyDeopt()852 void CodeGenerator::EnsureSpaceForLazyDeopt() {
853 int space_needed = Deoptimizer::patch_size();
854 if (!linkage()->info()->IsStub()) {
855 // Ensure that we have enough space after the previous lazy-bailout
856 // instruction for patching the code here.
857 int current_pc = masm()->pc_offset();
858 if (current_pc < last_lazy_deopt_pc_ + space_needed) {
859 // Block literal pool emission for duration of padding.
860 v8::internal::Assembler::BlockConstPoolScope block_const_pool(masm());
861 int padding_size = last_lazy_deopt_pc_ + space_needed - current_pc;
862 DCHECK_EQ(0, padding_size % v8::internal::Assembler::kInstrSize);
863 while (padding_size > 0) {
864 __ nop();
865 padding_size -= v8::internal::Assembler::kInstrSize;
866 }
867 }
868 }
869 MarkLazyDeoptSite();
870 }
871
872 #undef __
873
874 } // namespace compiler
875 } // namespace internal
876 } // namespace v8
877