Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(620)

Side by Side Diff: src/ppc/assembler-ppc.h

Issue 571173003: PowerPC specific sub-directories (Closed) Base URL: http://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Updated ppc sub-dirs to current V8 code levels Created 6 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 1994-2006 Sun Microsystems Inc.
2 // All Rights Reserved.
3 //
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions
6 // are met:
7 //
8 // - Redistributions of source code must retain the above copyright notice,
9 // this list of conditions and the following disclaimer.
10 //
11 // - Redistribution in binary form must reproduce the above copyright
12 // notice, this list of conditions and the following disclaimer in the
13 // documentation and/or other materials provided with the
14 // distribution.
15 //
16 // - Neither the name of Sun Microsystems or the names of contributors may
17 // be used to endorse or promote products derived from this software without
18 // specific prior written permission.
19 //
20 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23 // FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
24 // COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25 // INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
26 // (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
27 // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28 // HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
29 // STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
30 // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
31 // OF THE POSSIBILITY OF SUCH DAMAGE.
32
33 // The original source code covered by the above license above has been
34 // modified significantly by Google Inc.
35 // Copyright 2012 the V8 project authors. All rights reserved.
36
37 //
38 // Copyright IBM Corp. 2012, 2013. All rights reserved.
39 //
40
41 // A light-weight PPC Assembler
42 // Generates user mode instructions for the PPC architecture up
43
44 #ifndef V8_PPC_ASSEMBLER_PPC_H_
45 #define V8_PPC_ASSEMBLER_PPC_H_
46
47 #include <stdio.h>
48 #include <vector>
49
50 #include "src/assembler.h"
51 #include "src/ppc/constants-ppc.h"
52 #include "src/serialize.h"
53
54 #define ABI_USES_FUNCTION_DESCRIPTORS \
55 (V8_HOST_ARCH_PPC&&(V8_OS_AIX || \
56 (V8_TARGET_ARCH_PPC64 && V8_TARGET_BIG_ENDIAN)))
57
58 #define ABI_PASSES_HANDLES_IN_REGS \
59 (!V8_HOST_ARCH_PPC || V8_OS_AIX || V8_TARGET_ARCH_PPC64)
60
61 #define ABI_RETURNS_HANDLES_IN_REGS \
62 (!V8_HOST_ARCH_PPC || V8_TARGET_LITTLE_ENDIAN)
63
64 #define ABI_RETURNS_OBJECT_PAIRS_IN_REGS \
65 (!V8_HOST_ARCH_PPC || V8_TARGET_LITTLE_ENDIAN)
66
67 #define ABI_TOC_ADDRESSABILITY_VIA_IP \
68 (V8_HOST_ARCH_PPC && V8_TARGET_ARCH_PPC64 && V8_TARGET_LITTLE_ENDIAN)
69
70 #if !V8_HOST_ARCH_PPC || V8_OS_AIX || V8_TARGET_ARCH_PPC64
71 #define ABI_TOC_REGISTER kRegister_r2_Code
72 #else
73 #define ABI_TOC_REGISTER kRegister_r13_Code
74 #endif
75
76 #define INSTR_AND_DATA_CACHE_COHERENCY LWSYNC
77
78 namespace v8 {
79 namespace internal {
80
81 // CPU Registers.
82 //
83 // 1) We would prefer to use an enum, but enum values are assignment-
84 // compatible with int, which has caused code-generation bugs.
85 //
86 // 2) We would prefer to use a class instead of a struct but we don't like
87 // the register initialization to depend on the particular initialization
88 // order (which appears to be different on OS X, Linux, and Windows for the
89 // installed versions of C++ we tried). Using a struct permits C-style
90 // "initialization". Also, the Register objects cannot be const as this
91 // forces initialization stubs in MSVC, making us dependent on initialization
92 // order.
93 //
94 // 3) By not using an enum, we are possibly preventing the compiler from
95 // doing certain constant folds, which may significantly reduce the
96 // code generated for some assembly instructions (because they boil down
97 // to a few constants). If this is a problem, we could change the code
98 // such that we use an enum in optimized mode, and the struct in debug
99 // mode. This way we get the compile-time error checking in debug mode
100 // and best performance in optimized code.
101
102 // Core register
103 struct Register {
104 static const int kNumRegisters = 32;
105 static const int kSizeInBytes = kPointerSize;
106
107 #if V8_TARGET_LITTLE_ENDIAN
108 static const int kMantissaOffset = 0;
109 static const int kExponentOffset = 4;
110 #else
111 static const int kMantissaOffset = 4;
112 static const int kExponentOffset = 0;
113 #endif
114
115 static const int kAllocatableLowRangeBegin = 3;
116 static const int kAllocatableLowRangeEnd = 10;
117 static const int kAllocatableHighRangeBegin = 14;
118 #if V8_OOL_CONSTANT_POOL
119 static const int kAllocatableHighRangeEnd = 27;
120 #else
121 static const int kAllocatableHighRangeEnd = 28;
122 #endif
123 static const int kAllocatableContext = 30;
124
125 static const int kNumAllocatableLow =
126 kAllocatableLowRangeEnd - kAllocatableLowRangeBegin + 1;
127 static const int kNumAllocatableHigh =
128 kAllocatableHighRangeEnd - kAllocatableHighRangeBegin + 1;
129 static const int kMaxNumAllocatableRegisters =
130 kNumAllocatableLow + kNumAllocatableHigh + 1; // cp
131 static int NumAllocatableRegisters() { return kMaxNumAllocatableRegisters; }
132
133 static int ToAllocationIndex(Register reg) {
134 int index;
135 int code = reg.code();
136 if (code == kAllocatableContext) {
137 // Context is the last index
138 index = NumAllocatableRegisters() - 1;
139 } else if (code <= kAllocatableLowRangeEnd) {
140 // low range
141 index = code - kAllocatableLowRangeBegin;
142 } else {
143 // high range
144 index = code - kAllocatableHighRangeBegin + kNumAllocatableLow;
145 }
146 DCHECK(index >= 0 && index < kMaxNumAllocatableRegisters);
147 return index;
148 }
149
150 static Register FromAllocationIndex(int index) {
151 DCHECK(index >= 0 && index < kMaxNumAllocatableRegisters);
152 // Last index is always the 'cp' register.
153 if (index == kMaxNumAllocatableRegisters - 1) {
154 return from_code(kAllocatableContext);
155 }
156 return (index < kNumAllocatableLow)
157 ? from_code(index + kAllocatableLowRangeBegin)
158 : from_code(index - kNumAllocatableLow +
159 kAllocatableHighRangeBegin);
160 }
161
162 static const char* AllocationIndexToString(int index) {
163 DCHECK(index >= 0 && index < kMaxNumAllocatableRegisters);
164 const char* const names[] = {
165 "r3",
166 "r4",
167 "r5",
168 "r6",
169 "r7",
170 "r8",
171 "r9",
172 "r10",
173 "r14",
174 "r15",
175 "r16",
176 "r17",
177 "r18",
178 "r19",
179 "r20",
180 "r21",
181 "r22",
182 "r23",
183 "r24",
184 "r25",
185 "r26",
186 "r27",
187 #if !V8_OOL_CONSTANT_POOL
188 "r28",
189 #endif
190 "cp",
191 };
192 return names[index];
193 }
194
195 static Register from_code(int code) {
196 Register r = {code};
197 return r;
198 }
199
200 bool is_valid() const { return 0 <= code_ && code_ < kNumRegisters; }
201 bool is(Register reg) const { return code_ == reg.code_; }
202 int code() const {
203 DCHECK(is_valid());
204 return code_;
205 }
206 int bit() const {
207 DCHECK(is_valid());
208 return 1 << code_;
209 }
210
211 void set_code(int code) {
212 code_ = code;
213 DCHECK(is_valid());
214 }
215
216 // Unfortunately we can't make this private in a struct.
217 int code_;
218 };
219
220 // These constants are used in several locations, including static initializers
221 const int kRegister_no_reg_Code = -1;
222 const int kRegister_r0_Code = 0; // general scratch
223 const int kRegister_sp_Code = 1; // stack pointer
224 const int kRegister_r2_Code = 2; // special on PowerPC
225 const int kRegister_r3_Code = 3;
226 const int kRegister_r4_Code = 4;
227 const int kRegister_r5_Code = 5;
228 const int kRegister_r6_Code = 6;
229 const int kRegister_r7_Code = 7;
230 const int kRegister_r8_Code = 8;
231 const int kRegister_r9_Code = 9;
232 const int kRegister_r10_Code = 10;
233 const int kRegister_r11_Code = 11; // lithium scratch
234 const int kRegister_ip_Code = 12; // ip (general scratch)
235 const int kRegister_r13_Code = 13; // special on PowerPC
236 const int kRegister_r14_Code = 14;
237 const int kRegister_r15_Code = 15;
238
239 const int kRegister_r16_Code = 16;
240 const int kRegister_r17_Code = 17;
241 const int kRegister_r18_Code = 18;
242 const int kRegister_r19_Code = 19;
243 const int kRegister_r20_Code = 20;
244 const int kRegister_r21_Code = 21;
245 const int kRegister_r22_Code = 22;
246 const int kRegister_r23_Code = 23;
247 const int kRegister_r24_Code = 24;
248 const int kRegister_r25_Code = 25;
249 const int kRegister_r26_Code = 26;
250 const int kRegister_r27_Code = 27;
251 const int kRegister_r28_Code = 28; // constant pool pointer
252 const int kRegister_r29_Code = 29; // roots array pointer
253 const int kRegister_r30_Code = 30; // context pointer
254 const int kRegister_fp_Code = 31; // frame pointer
255
256 const Register no_reg = {kRegister_no_reg_Code};
257
258 const Register r0 = {kRegister_r0_Code};
259 const Register sp = {kRegister_sp_Code};
260 const Register r2 = {kRegister_r2_Code};
261 const Register r3 = {kRegister_r3_Code};
262 const Register r4 = {kRegister_r4_Code};
263 const Register r5 = {kRegister_r5_Code};
264 const Register r6 = {kRegister_r6_Code};
265 const Register r7 = {kRegister_r7_Code};
266 const Register r8 = {kRegister_r8_Code};
267 const Register r9 = {kRegister_r9_Code};
268 const Register r10 = {kRegister_r10_Code};
269 const Register r11 = {kRegister_r11_Code};
270 const Register ip = {kRegister_ip_Code};
271 const Register r13 = {kRegister_r13_Code};
272 const Register r14 = {kRegister_r14_Code};
273 const Register r15 = {kRegister_r15_Code};
274
275 const Register r16 = {kRegister_r16_Code};
276 const Register r17 = {kRegister_r17_Code};
277 const Register r18 = {kRegister_r18_Code};
278 const Register r19 = {kRegister_r19_Code};
279 const Register r20 = {kRegister_r20_Code};
280 const Register r21 = {kRegister_r21_Code};
281 const Register r22 = {kRegister_r22_Code};
282 const Register r23 = {kRegister_r23_Code};
283 const Register r24 = {kRegister_r24_Code};
284 const Register r25 = {kRegister_r25_Code};
285 const Register r26 = {kRegister_r26_Code};
286 const Register r27 = {kRegister_r27_Code};
287 const Register r28 = {kRegister_r28_Code};
288 const Register r29 = {kRegister_r29_Code};
289 const Register r30 = {kRegister_r30_Code};
290 const Register fp = {kRegister_fp_Code};
291
292 // Give alias names to registers
293 const Register cp = {kRegister_r30_Code}; // JavaScript context pointer
294 const Register kRootRegister = {kRegister_r29_Code}; // Roots array pointer.
295 #if V8_OOL_CONSTANT_POOL
296 const Register kConstantPoolRegister = {kRegister_r28_Code}; // Constant pool
297 #endif
298
299 // Double word FP register.
300 struct DoubleRegister {
301 static const int kNumRegisters = 32;
302 static const int kMaxNumRegisters = kNumRegisters;
303 static const int kNumVolatileRegisters = 14; // d0-d13
304 static const int kSizeInBytes = 8;
305
306 static const int kAllocatableLowRangeBegin = 1;
307 static const int kAllocatableLowRangeEnd = 12;
308 static const int kAllocatableHighRangeBegin = 15;
309 static const int kAllocatableHighRangeEnd = 31;
310
311 static const int kNumAllocatableLow =
312 kAllocatableLowRangeEnd - kAllocatableLowRangeBegin + 1;
313 static const int kNumAllocatableHigh =
314 kAllocatableHighRangeEnd - kAllocatableHighRangeBegin + 1;
315 static const int kMaxNumAllocatableRegisters =
316 kNumAllocatableLow + kNumAllocatableHigh;
317 static int NumAllocatableRegisters() { return kMaxNumAllocatableRegisters; }
318
319 static int ToAllocationIndex(DoubleRegister reg) {
320 int code = reg.code();
321 int index = (code <= kAllocatableLowRangeEnd)
322 ? code - kAllocatableLowRangeBegin
323 : code - kAllocatableHighRangeBegin + kNumAllocatableLow;
324 DCHECK(index < kMaxNumAllocatableRegisters);
325 return index;
326 }
327
328 static DoubleRegister FromAllocationIndex(int index) {
329 DCHECK(index >= 0 && index < kMaxNumAllocatableRegisters);
330 return (index < kNumAllocatableLow)
331 ? from_code(index + kAllocatableLowRangeBegin)
332 : from_code(index - kNumAllocatableLow +
333 kAllocatableHighRangeBegin);
334 }
335
336 static const char* AllocationIndexToString(int index);
337
338 static DoubleRegister from_code(int code) {
339 DoubleRegister r = {code};
340 return r;
341 }
342
343 bool is_valid() const { return 0 <= code_ && code_ < kMaxNumRegisters; }
344 bool is(DoubleRegister reg) const { return code_ == reg.code_; }
345
346 int code() const {
347 DCHECK(is_valid());
348 return code_;
349 }
350 int bit() const {
351 DCHECK(is_valid());
352 return 1 << code_;
353 }
354 void split_code(int* vm, int* m) const {
355 DCHECK(is_valid());
356 *m = (code_ & 0x10) >> 4;
357 *vm = code_ & 0x0F;
358 }
359
360 int code_;
361 };
362
363
364 const DoubleRegister no_dreg = {-1};
365 const DoubleRegister d0 = {0};
366 const DoubleRegister d1 = {1};
367 const DoubleRegister d2 = {2};
368 const DoubleRegister d3 = {3};
369 const DoubleRegister d4 = {4};
370 const DoubleRegister d5 = {5};
371 const DoubleRegister d6 = {6};
372 const DoubleRegister d7 = {7};
373 const DoubleRegister d8 = {8};
374 const DoubleRegister d9 = {9};
375 const DoubleRegister d10 = {10};
376 const DoubleRegister d11 = {11};
377 const DoubleRegister d12 = {12};
378 const DoubleRegister d13 = {13};
379 const DoubleRegister d14 = {14};
380 const DoubleRegister d15 = {15};
381 const DoubleRegister d16 = {16};
382 const DoubleRegister d17 = {17};
383 const DoubleRegister d18 = {18};
384 const DoubleRegister d19 = {19};
385 const DoubleRegister d20 = {20};
386 const DoubleRegister d21 = {21};
387 const DoubleRegister d22 = {22};
388 const DoubleRegister d23 = {23};
389 const DoubleRegister d24 = {24};
390 const DoubleRegister d25 = {25};
391 const DoubleRegister d26 = {26};
392 const DoubleRegister d27 = {27};
393 const DoubleRegister d28 = {28};
394 const DoubleRegister d29 = {29};
395 const DoubleRegister d30 = {30};
396 const DoubleRegister d31 = {31};
397
398 // Aliases for double registers. Defined using #define instead of
399 // "static const DoubleRegister&" because Clang complains otherwise when a
400 // compilation unit that includes this header doesn't use the variables.
401 #define kFirstCalleeSavedDoubleReg d14
402 #define kLastCalleeSavedDoubleReg d31
403 #define kDoubleRegZero d14
404 #define kScratchDoubleReg d13
405
406 Register ToRegister(int num);
407
408 // Coprocessor register
409 struct CRegister {
410 bool is_valid() const { return 0 <= code_ && code_ < 16; }
411 bool is(CRegister creg) const { return code_ == creg.code_; }
412 int code() const {
413 DCHECK(is_valid());
414 return code_;
415 }
416 int bit() const {
417 DCHECK(is_valid());
418 return 1 << code_;
419 }
420
421 // Unfortunately we can't make this private in a struct.
422 int code_;
423 };
424
425
426 const CRegister no_creg = {-1};
427
428 const CRegister cr0 = {0};
429 const CRegister cr1 = {1};
430 const CRegister cr2 = {2};
431 const CRegister cr3 = {3};
432 const CRegister cr4 = {4};
433 const CRegister cr5 = {5};
434 const CRegister cr6 = {6};
435 const CRegister cr7 = {7};
436 const CRegister cr8 = {8};
437 const CRegister cr9 = {9};
438 const CRegister cr10 = {10};
439 const CRegister cr11 = {11};
440 const CRegister cr12 = {12};
441 const CRegister cr13 = {13};
442 const CRegister cr14 = {14};
443 const CRegister cr15 = {15};
444
445 // -----------------------------------------------------------------------------
446 // Machine instruction Operands
447
448 #if V8_TARGET_ARCH_PPC64
449 const RelocInfo::Mode kRelocInfo_NONEPTR = RelocInfo::NONE64;
450 #else
451 const RelocInfo::Mode kRelocInfo_NONEPTR = RelocInfo::NONE32;
452 #endif
453
454 // Class Operand represents a shifter operand in data processing instructions
455 class Operand BASE_EMBEDDED {
456 public:
457 // immediate
458 INLINE(explicit Operand(intptr_t immediate,
459 RelocInfo::Mode rmode = kRelocInfo_NONEPTR));
460 INLINE(static Operand Zero()) { return Operand(static_cast<intptr_t>(0)); }
461 INLINE(explicit Operand(const ExternalReference& f));
462 explicit Operand(Handle<Object> handle);
463 INLINE(explicit Operand(Smi* value));
464
465 // rm
466 INLINE(explicit Operand(Register rm));
467
468 // Return true if this is a register operand.
469 INLINE(bool is_reg() const);
470
471 // For mov. Return the number of actual instructions required to
472 // load the operand into a register. This can be anywhere from
473 // one (constant pool small section) to five instructions (full
474 // 64-bit sequence).
475 //
476 // The value returned is only valid as long as no entries are added to the
477 // constant pool between this call and the actual instruction being emitted.
478 bool must_output_reloc_info(const Assembler* assembler) const;
479
480 inline intptr_t immediate() const {
481 DCHECK(!rm_.is_valid());
482 return imm_;
483 }
484
485 Register rm() const { return rm_; }
486
487 private:
488 Register rm_;
489 intptr_t imm_; // valid if rm_ == no_reg
490 RelocInfo::Mode rmode_;
491
492 friend class Assembler;
493 friend class MacroAssembler;
494 };
495
496
497 // Class MemOperand represents a memory operand in load and store instructions
498 // On PowerPC we have base register + 16bit signed value
499 // Alternatively we can have a 16bit signed value immediate
500 class MemOperand BASE_EMBEDDED {
501 public:
502 explicit MemOperand(Register rn, int32_t offset = 0);
503
504 explicit MemOperand(Register ra, Register rb);
505
506 int32_t offset() const {
507 DCHECK(rb_.is(no_reg));
508 return offset_;
509 }
510
511 // PowerPC - base register
512 Register ra() const {
513 DCHECK(!ra_.is(no_reg));
514 return ra_;
515 }
516
517 Register rb() const {
518 DCHECK(offset_ == 0 && !rb_.is(no_reg));
519 return rb_;
520 }
521
522 private:
523 Register ra_; // base
524 int32_t offset_; // offset
525 Register rb_; // index
526
527 friend class Assembler;
528 };
529
530
531 #if V8_OOL_CONSTANT_POOL
532 // Class used to build a constant pool.
533 class ConstantPoolBuilder BASE_EMBEDDED {
534 public:
535 ConstantPoolBuilder();
536 ConstantPoolArray::LayoutSection AddEntry(Assembler* assm,
537 const RelocInfo& rinfo);
538 void Relocate(intptr_t pc_delta);
539 bool IsEmpty();
540 Handle<ConstantPoolArray> New(Isolate* isolate);
541 void Populate(Assembler* assm, ConstantPoolArray* constant_pool);
542
543 inline ConstantPoolArray::LayoutSection current_section() const {
544 return current_section_;
545 }
546
547 // Rather than increasing the capacity of the ConstantPoolArray's
548 // small section to match the longer (16-bit) reach of PPC's load
549 // instruction (at the expense of a larger header to describe the
550 // layout), the PPC implementation utilizes the extended section to
551 // satisfy that reach. I.e. all entries (regardless of their
552 // section) are reachable with a single load instruction.
553 //
554 // This implementation does not support an unlimited constant pool
555 // size (which would require a multi-instruction sequence). [See
556 // ARM commit e27ab337 for a reference on the changes required to
557 // support the longer instruction sequence.] Note, however, that
558 // going down that path will necessarily generate that longer
559 // sequence for all extended section accesses since the placement of
560 // a given entry within the section is not known at the time of
561 // code generation.
562 //
563 // TODO(mbrandy): Determine whether there is a benefit to supporting
564 // the longer sequence given that nops could be used for those
565 // entries which are reachable with a single instruction.
566 inline bool is_full() const { return !is_int16(size_); }
567
568 inline ConstantPoolArray::NumberOfEntries* number_of_entries(
569 ConstantPoolArray::LayoutSection section) {
570 return &number_of_entries_[section];
571 }
572
573 inline ConstantPoolArray::NumberOfEntries* small_entries() {
574 return number_of_entries(ConstantPoolArray::SMALL_SECTION);
575 }
576
577 inline ConstantPoolArray::NumberOfEntries* extended_entries() {
578 return number_of_entries(ConstantPoolArray::EXTENDED_SECTION);
579 }
580
581 private:
582 struct ConstantPoolEntry {
583 ConstantPoolEntry(RelocInfo rinfo, ConstantPoolArray::LayoutSection section,
584 int merged_index)
585 : rinfo_(rinfo), section_(section), merged_index_(merged_index) {}
586
587 RelocInfo rinfo_;
588 ConstantPoolArray::LayoutSection section_;
589 int merged_index_;
590 };
591
592 ConstantPoolArray::Type GetConstantPoolType(RelocInfo::Mode rmode);
593
594 uint32_t size_;
595 std::vector<ConstantPoolEntry> entries_;
596 ConstantPoolArray::LayoutSection current_section_;
597 ConstantPoolArray::NumberOfEntries number_of_entries_[2];
598 };
599 #endif
600
601
602 class Assembler : public AssemblerBase {
603 public:
604 // Create an assembler. Instructions and relocation information are emitted
605 // into a buffer, with the instructions starting from the beginning and the
606 // relocation information starting from the end of the buffer. See CodeDesc
607 // for a detailed comment on the layout (globals.h).
608 //
609 // If the provided buffer is NULL, the assembler allocates and grows its own
610 // buffer, and buffer_size determines the initial buffer size. The buffer is
611 // owned by the assembler and deallocated upon destruction of the assembler.
612 //
613 // If the provided buffer is not NULL, the assembler uses the provided buffer
614 // for code generation and assumes its size to be buffer_size. If the buffer
615 // is too small, a fatal error occurs. No deallocation of the buffer is done
616 // upon destruction of the assembler.
617 Assembler(Isolate* isolate, void* buffer, int buffer_size);
618 virtual ~Assembler() {}
619
620 // GetCode emits any pending (non-emitted) code and fills the descriptor
621 // desc. GetCode() is idempotent; it returns the same result if no other
622 // Assembler functions are invoked in between GetCode() calls.
623 void GetCode(CodeDesc* desc);
624
625 // Label operations & relative jumps (PPUM Appendix D)
626 //
627 // Takes a branch opcode (cc) and a label (L) and generates
628 // either a backward branch or a forward branch and links it
629 // to the label fixup chain. Usage:
630 //
631 // Label L; // unbound label
632 // j(cc, &L); // forward branch to unbound label
633 // bind(&L); // bind label to the current pc
634 // j(cc, &L); // backward branch to bound label
635 // bind(&L); // illegal: a label may be bound only once
636 //
637 // Note: The same Label can be used for forward and backward branches
638 // but it may be bound only once.
639
640 void bind(Label* L); // binds an unbound label L to the current code position
641 // Determines if Label is bound and near enough so that a single
642 // branch instruction can be used to reach it.
643 bool is_near(Label* L, Condition cond);
644
645 // Returns the branch offset to the given label from the current code position
646 // Links the label to the current position if it is still unbound
647 // Manages the jump elimination optimization if the second parameter is true.
648 int branch_offset(Label* L, bool jump_elimination_allowed);
649
650 // Puts a labels target address at the given position.
651 // The high 8 bits are set to zero.
652 void label_at_put(Label* L, int at_offset);
653
654 #if V8_OOL_CONSTANT_POOL
655 INLINE(static bool IsConstantPoolLoadStart(Address pc));
656 INLINE(static bool IsConstantPoolLoadEnd(Address pc));
657 INLINE(static int GetConstantPoolOffset(Address pc));
658 INLINE(static void SetConstantPoolOffset(Address pc, int offset));
659
660 // Return the address in the constant pool of the code target address used by
661 // the branch/call instruction at pc, or the object in a mov.
662 INLINE(static Address target_constant_pool_address_at(
663 Address pc, ConstantPoolArray* constant_pool));
664 #endif
665
666 // Read/Modify the code target address in the branch/call instruction at pc.
667 INLINE(static Address target_address_at(Address pc,
668 ConstantPoolArray* constant_pool));
669 INLINE(static void set_target_address_at(
670 Address pc, ConstantPoolArray* constant_pool, Address target,
671 ICacheFlushMode icache_flush_mode = FLUSH_ICACHE_IF_NEEDED));
672 INLINE(static Address target_address_at(Address pc, Code* code)) {
673 ConstantPoolArray* constant_pool = code ? code->constant_pool() : NULL;
674 return target_address_at(pc, constant_pool);
675 }
676 INLINE(static void set_target_address_at(
677 Address pc, Code* code, Address target,
678 ICacheFlushMode icache_flush_mode = FLUSH_ICACHE_IF_NEEDED)) {
679 ConstantPoolArray* constant_pool = code ? code->constant_pool() : NULL;
680 set_target_address_at(pc, constant_pool, target, icache_flush_mode);
681 }
682
683 // Return the code target address at a call site from the return address
684 // of that call in the instruction stream.
685 inline static Address target_address_from_return_address(Address pc);
686
687 // Given the address of the beginning of a call, return the address
688 // in the instruction stream that the call will return to.
689 INLINE(static Address return_address_from_call_start(Address pc));
690
691 // Return the code target address of the patch debug break slot
692 INLINE(static Address break_address_from_return_address(Address pc));
693
694 // This sets the branch destination.
695 // This is for calls and branches within generated code.
696 inline static void deserialization_set_special_target_at(
697 Address instruction_payload, Code* code, Address target);
698
699 // Size of an instruction.
700 static const int kInstrSize = sizeof(Instr);
701
702 // Here we are patching the address in the LUI/ORI instruction pair.
703 // These values are used in the serialization process and must be zero for
704 // PPC platform, as Code, Embedded Object or External-reference pointers
705 // are split across two consecutive instructions and don't exist separately
706 // in the code, so the serializer should not step forwards in memory after
707 // a target is resolved and written.
708 static const int kSpecialTargetSize = 0;
709
710 // Number of instructions to load an address via a mov sequence.
711 #if V8_TARGET_ARCH_PPC64
712 static const int kMovInstructionsConstantPool = 2;
713 static const int kMovInstructionsNoConstantPool = 5;
714 #else
715 static const int kMovInstructionsConstantPool = 1;
716 static const int kMovInstructionsNoConstantPool = 2;
717 #endif
718 #if V8_OOL_CONSTANT_POOL
719 static const int kMovInstructions = kMovInstructionsConstantPool;
720 #else
721 static const int kMovInstructions = kMovInstructionsNoConstantPool;
722 #endif
723
724 // Distance between the instruction referring to the address of the call
725 // target and the return address.
726
727 // Call sequence is a FIXED_SEQUENCE:
728 // mov r8, @ call address
729 // mtlr r8
730 // blrl
731 // @ return address
732 static const int kCallTargetAddressOffset =
733 (kMovInstructions + 2) * kInstrSize;
734
735 // Distance between start of patched return sequence and the emitted address
736 // to jump to.
737 // Patched return sequence is a FIXED_SEQUENCE:
738 // mov r0, <address>
739 // mtlr r0
740 // blrl
741 static const int kPatchReturnSequenceAddressOffset = 0 * kInstrSize;
742
743 // Distance between start of patched debug break slot and the emitted address
744 // to jump to.
745 // Patched debug break slot code is a FIXED_SEQUENCE:
746 // mov r0, <address>
747 // mtlr r0
748 // blrl
749 static const int kPatchDebugBreakSlotAddressOffset = 0 * kInstrSize;
750
751 // This is the length of the BreakLocationIterator::SetDebugBreakAtReturn()
752 // code patch FIXED_SEQUENCE
753 static const int kJSReturnSequenceInstructions =
754 kMovInstructionsNoConstantPool + 3;
755
756 // This is the length of the code sequence from SetDebugBreakAtSlot()
757 // FIXED_SEQUENCE
758 static const int kDebugBreakSlotInstructions =
759 kMovInstructionsNoConstantPool + 2;
760 static const int kDebugBreakSlotLength =
761 kDebugBreakSlotInstructions * kInstrSize;
762
763 static inline int encode_crbit(const CRegister& cr, enum CRBit crbit) {
764 return ((cr.code() * CRWIDTH) + crbit);
765 }
766
767 // ---------------------------------------------------------------------------
768 // Code generation
769
770 // Insert the smallest number of nop instructions
771 // possible to align the pc offset to a multiple
772 // of m. m must be a power of 2 (>= 4).
773 void Align(int m);
774 // Aligns code to something that's optimal for a jump target for the platform.
775 void CodeTargetAlign();
776
777 // Branch instructions
778 void bclr(BOfield bo, LKBit lk);
779 void blr();
780 void bc(int branch_offset, BOfield bo, int condition_bit, LKBit lk = LeaveLK);
781 void b(int branch_offset, LKBit lk);
782
783 void bcctr(BOfield bo, LKBit lk);
784 void bctr();
785
786 // Convenience branch instructions using labels
787 void b(Label* L, LKBit lk = LeaveLK) { b(branch_offset(L, false), lk); }
788
789 void bc_short(Condition cond, Label* L, CRegister cr = cr7,
790 LKBit lk = LeaveLK) {
791 DCHECK(cond != al);
792 DCHECK(cr.code() >= 0 && cr.code() <= 7);
793
794 int b_offset = branch_offset(L, false);
795
796 switch (cond) {
797 case eq:
798 bc(b_offset, BT, encode_crbit(cr, CR_EQ), lk);
799 break;
800 case ne:
801 bc(b_offset, BF, encode_crbit(cr, CR_EQ), lk);
802 break;
803 case gt:
804 bc(b_offset, BT, encode_crbit(cr, CR_GT), lk);
805 break;
806 case le:
807 bc(b_offset, BF, encode_crbit(cr, CR_GT), lk);
808 break;
809 case lt:
810 bc(b_offset, BT, encode_crbit(cr, CR_LT), lk);
811 break;
812 case ge:
813 bc(b_offset, BF, encode_crbit(cr, CR_LT), lk);
814 break;
815 case unordered:
816 bc(b_offset, BT, encode_crbit(cr, CR_FU), lk);
817 break;
818 case ordered:
819 bc(b_offset, BF, encode_crbit(cr, CR_FU), lk);
820 break;
821 case overflow:
822 bc(b_offset, BT, encode_crbit(cr, CR_SO), lk);
823 break;
824 case nooverflow:
825 bc(b_offset, BF, encode_crbit(cr, CR_SO), lk);
826 break;
827 default:
828 UNIMPLEMENTED();
829 }
830 }
831
832 void b(Condition cond, Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
833 if (cond == al) {
834 b(L, lk);
835 return;
836 }
837
838 if ((L->is_bound() && is_near(L, cond)) || !is_trampoline_emitted()) {
839 bc_short(cond, L, cr, lk);
840 return;
841 }
842
843 Label skip;
844 Condition neg_cond = NegateCondition(cond);
845 bc_short(neg_cond, &skip, cr);
846 b(L, lk);
847 bind(&skip);
848 }
849
850 void bne(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
851 b(ne, L, cr, lk);
852 }
853 void beq(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
854 b(eq, L, cr, lk);
855 }
856 void blt(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
857 b(lt, L, cr, lk);
858 }
859 void bge(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
860 b(ge, L, cr, lk);
861 }
862 void ble(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
863 b(le, L, cr, lk);
864 }
865 void bgt(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
866 b(gt, L, cr, lk);
867 }
868 void bunordered(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
869 b(unordered, L, cr, lk);
870 }
871 void bordered(Label* L, CRegister cr = cr7, LKBit lk = LeaveLK) {
872 b(ordered, L, cr, lk);
873 }
874 void boverflow(Label* L, CRegister cr = cr0, LKBit lk = LeaveLK) {
875 b(overflow, L, cr, lk);
876 }
877 void bnooverflow(Label* L, CRegister cr = cr0, LKBit lk = LeaveLK) {
878 b(nooverflow, L, cr, lk);
879 }
880
881 // Decrement CTR; branch if CTR != 0
882 void bdnz(Label* L, LKBit lk = LeaveLK) {
883 bc(branch_offset(L, false), DCBNZ, 0, lk);
884 }
885
886 // Data-processing instructions
887
888 void sub(Register dst, Register src1, Register src2, OEBit s = LeaveOE,
889 RCBit r = LeaveRC);
890
891 void subfic(Register dst, Register src, const Operand& imm);
892
893 void subfc(Register dst, Register src1, Register src2, OEBit s = LeaveOE,
894 RCBit r = LeaveRC);
895
896 void add(Register dst, Register src1, Register src2, OEBit s = LeaveOE,
897 RCBit r = LeaveRC);
898
899 void addc(Register dst, Register src1, Register src2, OEBit o = LeaveOE,
900 RCBit r = LeaveRC);
901
902 void addze(Register dst, Register src1, OEBit o, RCBit r);
903
904 void mullw(Register dst, Register src1, Register src2, OEBit o = LeaveOE,
905 RCBit r = LeaveRC);
906
907 void mulhw(Register dst, Register src1, Register src2, OEBit o = LeaveOE,
908 RCBit r = LeaveRC);
909
910 void divw(Register dst, Register src1, Register src2, OEBit o = LeaveOE,
911 RCBit r = LeaveRC);
912
913 void addi(Register dst, Register src, const Operand& imm);
914 void addis(Register dst, Register src, const Operand& imm);
915 void addic(Register dst, Register src, const Operand& imm);
916
917 void and_(Register dst, Register src1, Register src2, RCBit rc = LeaveRC);
918 void andc(Register dst, Register src1, Register src2, RCBit rc = LeaveRC);
919 void andi(Register ra, Register rs, const Operand& imm);
920 void andis(Register ra, Register rs, const Operand& imm);
921 void nor(Register dst, Register src1, Register src2, RCBit r = LeaveRC);
922 void notx(Register dst, Register src, RCBit r = LeaveRC);
923 void ori(Register dst, Register src, const Operand& imm);
924 void oris(Register dst, Register src, const Operand& imm);
925 void orx(Register dst, Register src1, Register src2, RCBit rc = LeaveRC);
926 void xori(Register dst, Register src, const Operand& imm);
927 void xoris(Register ra, Register rs, const Operand& imm);
928 void xor_(Register dst, Register src1, Register src2, RCBit rc = LeaveRC);
929 void cmpi(Register src1, const Operand& src2, CRegister cr = cr7);
930 void cmpli(Register src1, const Operand& src2, CRegister cr = cr7);
931 void cmpwi(Register src1, const Operand& src2, CRegister cr = cr7);
932 void cmplwi(Register src1, const Operand& src2, CRegister cr = cr7);
933 void li(Register dst, const Operand& src);
934 void lis(Register dst, const Operand& imm);
935 void mr(Register dst, Register src);
936
937 void lbz(Register dst, const MemOperand& src);
938 void lbzx(Register dst, const MemOperand& src);
939 void lbzux(Register dst, const MemOperand& src);
940 void lhz(Register dst, const MemOperand& src);
941 void lhzx(Register dst, const MemOperand& src);
942 void lhzux(Register dst, const MemOperand& src);
943 void lwz(Register dst, const MemOperand& src);
944 void lwzu(Register dst, const MemOperand& src);
945 void lwzx(Register dst, const MemOperand& src);
946 void lwzux(Register dst, const MemOperand& src);
947 void lwa(Register dst, const MemOperand& src);
948 void stb(Register dst, const MemOperand& src);
949 void stbx(Register dst, const MemOperand& src);
950 void stbux(Register dst, const MemOperand& src);
951 void sth(Register dst, const MemOperand& src);
952 void sthx(Register dst, const MemOperand& src);
953 void sthux(Register dst, const MemOperand& src);
954 void stw(Register dst, const MemOperand& src);
955 void stwu(Register dst, const MemOperand& src);
956 void stwx(Register rs, const MemOperand& src);
957 void stwux(Register rs, const MemOperand& src);
958
959 void extsb(Register rs, Register ra, RCBit r = LeaveRC);
960 void extsh(Register rs, Register ra, RCBit r = LeaveRC);
961
962 void neg(Register rt, Register ra, OEBit o = LeaveOE, RCBit c = LeaveRC);
963
964 #if V8_TARGET_ARCH_PPC64
965 void ld(Register rd, const MemOperand& src);
966 void ldx(Register rd, const MemOperand& src);
967 void ldu(Register rd, const MemOperand& src);
968 void ldux(Register rd, const MemOperand& src);
969 void std(Register rs, const MemOperand& src);
970 void stdx(Register rs, const MemOperand& src);
971 void stdu(Register rs, const MemOperand& src);
972 void stdux(Register rs, const MemOperand& src);
973 void rldic(Register dst, Register src, int sh, int mb, RCBit r = LeaveRC);
974 void rldicl(Register dst, Register src, int sh, int mb, RCBit r = LeaveRC);
975 void rldcl(Register ra, Register rs, Register rb, int mb, RCBit r = LeaveRC);
976 void rldicr(Register dst, Register src, int sh, int me, RCBit r = LeaveRC);
977 void rldimi(Register dst, Register src, int sh, int mb, RCBit r = LeaveRC);
978 void sldi(Register dst, Register src, const Operand& val, RCBit rc = LeaveRC);
979 void srdi(Register dst, Register src, const Operand& val, RCBit rc = LeaveRC);
980 void clrrdi(Register dst, Register src, const Operand& val,
981 RCBit rc = LeaveRC);
982 void clrldi(Register dst, Register src, const Operand& val,
983 RCBit rc = LeaveRC);
984 void sradi(Register ra, Register rs, int sh, RCBit r = LeaveRC);
985 void srd(Register dst, Register src1, Register src2, RCBit r = LeaveRC);
986 void sld(Register dst, Register src1, Register src2, RCBit r = LeaveRC);
987 void srad(Register dst, Register src1, Register src2, RCBit r = LeaveRC);
988 void rotld(Register ra, Register rs, Register rb, RCBit r = LeaveRC);
989 void rotldi(Register ra, Register rs, int sh, RCBit r = LeaveRC);
990 void rotrdi(Register ra, Register rs, int sh, RCBit r = LeaveRC);
991 void cntlzd_(Register dst, Register src, RCBit rc = LeaveRC);
992 void extsw(Register rs, Register ra, RCBit r = LeaveRC);
993 void mulld(Register dst, Register src1, Register src2, OEBit o = LeaveOE,
994 RCBit r = LeaveRC);
995 void divd(Register dst, Register src1, Register src2, OEBit o = LeaveOE,
996 RCBit r = LeaveRC);
997 #endif
998
999 void rlwinm(Register ra, Register rs, int sh, int mb, int me,
1000 RCBit rc = LeaveRC);
1001 void rlwimi(Register ra, Register rs, int sh, int mb, int me,
1002 RCBit rc = LeaveRC);
1003 void rlwnm(Register ra, Register rs, Register rb, int mb, int me,
1004 RCBit rc = LeaveRC);
1005 void slwi(Register dst, Register src, const Operand& val, RCBit rc = LeaveRC);
1006 void srwi(Register dst, Register src, const Operand& val, RCBit rc = LeaveRC);
1007 void clrrwi(Register dst, Register src, const Operand& val,
1008 RCBit rc = LeaveRC);
1009 void clrlwi(Register dst, Register src, const Operand& val,
1010 RCBit rc = LeaveRC);
1011 void srawi(Register ra, Register rs, int sh, RCBit r = LeaveRC);
1012 void srw(Register dst, Register src1, Register src2, RCBit r = LeaveRC);
1013 void slw(Register dst, Register src1, Register src2, RCBit r = LeaveRC);
1014 void sraw(Register dst, Register src1, Register src2, RCBit r = LeaveRC);
1015 void rotlw(Register ra, Register rs, Register rb, RCBit r = LeaveRC);
1016 void rotlwi(Register ra, Register rs, int sh, RCBit r = LeaveRC);
1017 void rotrwi(Register ra, Register rs, int sh, RCBit r = LeaveRC);
1018
1019 void cntlzw_(Register dst, Register src, RCBit rc = LeaveRC);
1020
1021 void subi(Register dst, Register src1, const Operand& src2);
1022
1023 void cmp(Register src1, Register src2, CRegister cr = cr7);
1024 void cmpl(Register src1, Register src2, CRegister cr = cr7);
1025 void cmpw(Register src1, Register src2, CRegister cr = cr7);
1026 void cmplw(Register src1, Register src2, CRegister cr = cr7);
1027
1028 void mov(Register dst, const Operand& src);
1029
1030 // Load the position of the label relative to the generated code object
1031 // pointer in a register.
1032 void mov_label_offset(Register dst, Label* label);
1033
1034 // Multiply instructions
1035 void mul(Register dst, Register src1, Register src2, OEBit s = LeaveOE,
1036 RCBit r = LeaveRC);
1037
1038 // Miscellaneous arithmetic instructions
1039
1040 // Special register access
1041 void crxor(int bt, int ba, int bb);
1042 void crclr(int bt) { crxor(bt, bt, bt); }
1043 void creqv(int bt, int ba, int bb);
1044 void crset(int bt) { creqv(bt, bt, bt); }
1045 void mflr(Register dst);
1046 void mtlr(Register src);
1047 void mtctr(Register src);
1048 void mtxer(Register src);
1049 void mcrfs(int bf, int bfa);
1050 void mfcr(Register dst);
1051 #if V8_TARGET_ARCH_PPC64
1052 void mffprd(Register dst, DoubleRegister src);
1053 void mffprwz(Register dst, DoubleRegister src);
1054 void mtfprd(DoubleRegister dst, Register src);
1055 void mtfprwz(DoubleRegister dst, Register src);
1056 void mtfprwa(DoubleRegister dst, Register src);
1057 #endif
1058
1059 void fake_asm(enum FAKE_OPCODE_T fopcode);
1060 void marker_asm(int mcode);
1061 void function_descriptor();
1062
1063 // Exception-generating instructions and debugging support
1064 void stop(const char* msg, Condition cond = al,
1065 int32_t code = kDefaultStopCode, CRegister cr = cr7);
1066
1067 void bkpt(uint32_t imm16); // v5 and above
1068
1069 // Informational messages when simulating
1070 void info(const char* msg, Condition cond = al,
1071 int32_t code = kDefaultStopCode, CRegister cr = cr7);
1072
1073 void dcbf(Register ra, Register rb);
1074 void sync();
1075 void lwsync();
1076 void icbi(Register ra, Register rb);
1077 void isync();
1078
1079 // Support for floating point
1080 void lfd(const DoubleRegister frt, const MemOperand& src);
1081 void lfdu(const DoubleRegister frt, const MemOperand& src);
1082 void lfdx(const DoubleRegister frt, const MemOperand& src);
1083 void lfdux(const DoubleRegister frt, const MemOperand& src);
1084 void lfs(const DoubleRegister frt, const MemOperand& src);
1085 void lfsu(const DoubleRegister frt, const MemOperand& src);
1086 void lfsx(const DoubleRegister frt, const MemOperand& src);
1087 void lfsux(const DoubleRegister frt, const MemOperand& src);
1088 void stfd(const DoubleRegister frs, const MemOperand& src);
1089 void stfdu(const DoubleRegister frs, const MemOperand& src);
1090 void stfdx(const DoubleRegister frs, const MemOperand& src);
1091 void stfdux(const DoubleRegister frs, const MemOperand& src);
1092 void stfs(const DoubleRegister frs, const MemOperand& src);
1093 void stfsu(const DoubleRegister frs, const MemOperand& src);
1094 void stfsx(const DoubleRegister frs, const MemOperand& src);
1095 void stfsux(const DoubleRegister frs, const MemOperand& src);
1096
1097 void fadd(const DoubleRegister frt, const DoubleRegister fra,
1098 const DoubleRegister frb, RCBit rc = LeaveRC);
1099 void fsub(const DoubleRegister frt, const DoubleRegister fra,
1100 const DoubleRegister frb, RCBit rc = LeaveRC);
1101 void fdiv(const DoubleRegister frt, const DoubleRegister fra,
1102 const DoubleRegister frb, RCBit rc = LeaveRC);
1103 void fmul(const DoubleRegister frt, const DoubleRegister fra,
1104 const DoubleRegister frc, RCBit rc = LeaveRC);
1105 void fcmpu(const DoubleRegister fra, const DoubleRegister frb,
1106 CRegister cr = cr7);
1107 void fmr(const DoubleRegister frt, const DoubleRegister frb,
1108 RCBit rc = LeaveRC);
1109 void fctiwz(const DoubleRegister frt, const DoubleRegister frb);
1110 void fctiw(const DoubleRegister frt, const DoubleRegister frb);
1111 void frim(const DoubleRegister frt, const DoubleRegister frb);
1112 void frsp(const DoubleRegister frt, const DoubleRegister frb,
1113 RCBit rc = LeaveRC);
1114 void fcfid(const DoubleRegister frt, const DoubleRegister frb,
1115 RCBit rc = LeaveRC);
1116 void fctid(const DoubleRegister frt, const DoubleRegister frb,
1117 RCBit rc = LeaveRC);
1118 void fctidz(const DoubleRegister frt, const DoubleRegister frb,
1119 RCBit rc = LeaveRC);
1120 void fsel(const DoubleRegister frt, const DoubleRegister fra,
1121 const DoubleRegister frc, const DoubleRegister frb,
1122 RCBit rc = LeaveRC);
1123 void fneg(const DoubleRegister frt, const DoubleRegister frb,
1124 RCBit rc = LeaveRC);
1125 void mtfsfi(int bf, int immediate, RCBit rc = LeaveRC);
1126 void mffs(const DoubleRegister frt, RCBit rc = LeaveRC);
1127 void mtfsf(const DoubleRegister frb, bool L = 1, int FLM = 0, bool W = 0,
1128 RCBit rc = LeaveRC);
1129 void fsqrt(const DoubleRegister frt, const DoubleRegister frb,
1130 RCBit rc = LeaveRC);
1131 void fabs(const DoubleRegister frt, const DoubleRegister frb,
1132 RCBit rc = LeaveRC);
1133 void fmadd(const DoubleRegister frt, const DoubleRegister fra,
1134 const DoubleRegister frc, const DoubleRegister frb,
1135 RCBit rc = LeaveRC);
1136 void fmsub(const DoubleRegister frt, const DoubleRegister fra,
1137 const DoubleRegister frc, const DoubleRegister frb,
1138 RCBit rc = LeaveRC);
1139
1140 // Pseudo instructions
1141
1142 // Different nop operations are used by the code generator to detect certain
1143 // states of the generated code.
1144 enum NopMarkerTypes {
1145 NON_MARKING_NOP = 0,
1146 GROUP_ENDING_NOP,
1147 DEBUG_BREAK_NOP,
1148 // IC markers.
1149 PROPERTY_ACCESS_INLINED,
1150 PROPERTY_ACCESS_INLINED_CONTEXT,
1151 PROPERTY_ACCESS_INLINED_CONTEXT_DONT_DELETE,
1152 // Helper values.
1153 LAST_CODE_MARKER,
1154 FIRST_IC_MARKER = PROPERTY_ACCESS_INLINED
1155 };
1156
1157 void nop(int type = 0); // 0 is the default non-marking type.
1158
1159 void push(Register src) {
1160 #if V8_TARGET_ARCH_PPC64
1161 stdu(src, MemOperand(sp, -8));
1162 #else
1163 stwu(src, MemOperand(sp, -4));
1164 #endif
1165 }
1166
1167 void pop(Register dst) {
1168 #if V8_TARGET_ARCH_PPC64
1169 ld(dst, MemOperand(sp));
1170 addi(sp, sp, Operand(8));
1171 #else
1172 lwz(dst, MemOperand(sp));
1173 addi(sp, sp, Operand(4));
1174 #endif
1175 }
1176
1177 void pop() { addi(sp, sp, Operand(kPointerSize)); }
1178
1179 // Jump unconditionally to given label.
1180 void jmp(Label* L) { b(L); }
1181
1182 // Check the code size generated from label to here.
1183 int SizeOfCodeGeneratedSince(Label* label) {
1184 return pc_offset() - label->pos();
1185 }
1186
1187 // Check the number of instructions generated from label to here.
1188 int InstructionsGeneratedSince(Label* label) {
1189 return SizeOfCodeGeneratedSince(label) / kInstrSize;
1190 }
1191
1192 // Class for scoping postponing the trampoline pool generation.
1193 class BlockTrampolinePoolScope {
1194 public:
1195 explicit BlockTrampolinePoolScope(Assembler* assem) : assem_(assem) {
1196 assem_->StartBlockTrampolinePool();
1197 }
1198 ~BlockTrampolinePoolScope() { assem_->EndBlockTrampolinePool(); }
1199
1200 private:
1201 Assembler* assem_;
1202
1203 DISALLOW_IMPLICIT_CONSTRUCTORS(BlockTrampolinePoolScope);
1204 };
1205
1206 // Debugging
1207
1208 // Mark address of the ExitJSFrame code.
1209 void RecordJSReturn();
1210
1211 // Mark address of a debug break slot.
1212 void RecordDebugBreakSlot();
1213
1214 // Record the AST id of the CallIC being compiled, so that it can be placed
1215 // in the relocation information.
1216 void SetRecordedAstId(TypeFeedbackId ast_id) {
1217 // PPC - this shouldn't be failing roohack
1218 // DCHECK(recorded_ast_id_.IsNone());
1219 recorded_ast_id_ = ast_id;
1220 }
1221
1222 TypeFeedbackId RecordedAstId() {
1223 // roohack - another issue??? DCHECK(!recorded_ast_id_.IsNone());
1224 return recorded_ast_id_;
1225 }
1226
1227 void ClearRecordedAstId() { recorded_ast_id_ = TypeFeedbackId::None(); }
1228
1229 // Record a comment relocation entry that can be used by a disassembler.
1230 // Use --code-comments to enable.
1231 void RecordComment(const char* msg);
1232
1233 // Writes a single byte or word of data in the code stream. Used
1234 // for inline tables, e.g., jump-tables.
1235 void db(uint8_t data);
1236 void dd(uint32_t data);
1237 void emit_ptr(uintptr_t data);
1238
1239 PositionsRecorder* positions_recorder() { return &positions_recorder_; }
1240
1241 // Read/patch instructions
1242 Instr instr_at(int pos) { return *reinterpret_cast<Instr*>(buffer_ + pos); }
1243 void instr_at_put(int pos, Instr instr) {
1244 *reinterpret_cast<Instr*>(buffer_ + pos) = instr;
1245 }
1246 static Instr instr_at(byte* pc) { return *reinterpret_cast<Instr*>(pc); }
1247 static void instr_at_put(byte* pc, Instr instr) {
1248 *reinterpret_cast<Instr*>(pc) = instr;
1249 }
1250 static Condition GetCondition(Instr instr);
1251
1252 static bool IsLis(Instr instr);
1253 static bool IsLi(Instr instr);
1254 static bool IsAddic(Instr instr);
1255 static bool IsOri(Instr instr);
1256
1257 static bool IsBranch(Instr instr);
1258 static Register GetRA(Instr instr);
1259 static Register GetRB(Instr instr);
1260 #if V8_TARGET_ARCH_PPC64
1261 static bool Is64BitLoadIntoR12(Instr instr1, Instr instr2, Instr instr3,
1262 Instr instr4, Instr instr5);
1263 #else
1264 static bool Is32BitLoadIntoR12(Instr instr1, Instr instr2);
1265 #endif
1266
1267 static bool IsCmpRegister(Instr instr);
1268 static bool IsCmpImmediate(Instr instr);
1269 static bool IsRlwinm(Instr instr);
1270 #if V8_TARGET_ARCH_PPC64
1271 static bool IsRldicl(Instr instr);
1272 #endif
1273 static bool IsCrSet(Instr instr);
1274 static Register GetCmpImmediateRegister(Instr instr);
1275 static int GetCmpImmediateRawImmediate(Instr instr);
1276 static bool IsNop(Instr instr, int type = NON_MARKING_NOP);
1277
1278 // Postpone the generation of the trampoline pool for the specified number of
1279 // instructions.
1280 void BlockTrampolinePoolFor(int instructions);
1281 void CheckTrampolinePool();
1282
1283 int instructions_required_for_mov(const Operand& x) const;
1284
1285 #if V8_OOL_CONSTANT_POOL
1286 // Decide between using the constant pool vs. a mov immediate sequence.
1287 bool use_constant_pool_for_mov(const Operand& x, bool canOptimize) const;
1288
1289 // The code currently calls CheckBuffer() too often. This has the side
1290 // effect of randomly growing the buffer in the middle of multi-instruction
1291 // sequences.
1292 // MacroAssembler::LoadConstantPoolPointerRegister() includes a relocation
1293 // and multiple instructions. We cannot grow the buffer until the
1294 // relocation and all of the instructions are written.
1295 //
1296 // This function allows outside callers to check and grow the buffer
1297 void EnsureSpaceFor(int space_needed);
1298 #endif
1299
1300 // Allocate a constant pool of the correct size for the generated code.
1301 Handle<ConstantPoolArray> NewConstantPool(Isolate* isolate);
1302
1303 // Generate the constant pool for the generated code.
1304 void PopulateConstantPool(ConstantPoolArray* constant_pool);
1305
1306 #if V8_OOL_CONSTANT_POOL
1307 bool is_constant_pool_available() const { return constant_pool_available_; }
1308
1309 bool is_constant_pool_full() const {
1310 return constant_pool_builder_.is_full();
1311 }
1312
1313 bool use_extended_constant_pool() const {
1314 return constant_pool_builder_.current_section() ==
1315 ConstantPoolArray::EXTENDED_SECTION;
1316 }
1317 #endif
1318
1319 #if ABI_USES_FUNCTION_DESCRIPTORS || V8_OOL_CONSTANT_POOL
1320 static void RelocateInternalReference(
1321 Address pc, intptr_t delta, Address code_start,
1322 ICacheFlushMode icache_flush_mode = FLUSH_ICACHE_IF_NEEDED);
1323 static int DecodeInternalReference(Vector<char> buffer, Address pc);
1324 #endif
1325
1326 protected:
1327 // Relocation for a type-recording IC has the AST id added to it. This
1328 // member variable is a way to pass the information from the call site to
1329 // the relocation info.
1330 TypeFeedbackId recorded_ast_id_;
1331
1332 int buffer_space() const { return reloc_info_writer.pos() - pc_; }
1333
1334 // Decode branch instruction at pos and return branch target pos
1335 int target_at(int pos);
1336
1337 // Patch branch instruction at pos to branch to given branch target pos
1338 void target_at_put(int pos, int target_pos);
1339
1340 // Record reloc info for current pc_
1341 void RecordRelocInfo(RelocInfo::Mode rmode, intptr_t data = 0);
1342 void RecordRelocInfo(const RelocInfo& rinfo);
1343 #if V8_OOL_CONSTANT_POOL
1344 ConstantPoolArray::LayoutSection ConstantPoolAddEntry(
1345 const RelocInfo& rinfo) {
1346 return constant_pool_builder_.AddEntry(this, rinfo);
1347 }
1348 #endif
1349
1350 // Block the emission of the trampoline pool before pc_offset.
1351 void BlockTrampolinePoolBefore(int pc_offset) {
1352 if (no_trampoline_pool_before_ < pc_offset)
1353 no_trampoline_pool_before_ = pc_offset;
1354 }
1355
1356 void StartBlockTrampolinePool() { trampoline_pool_blocked_nesting_++; }
1357
1358 void EndBlockTrampolinePool() { trampoline_pool_blocked_nesting_--; }
1359
1360 bool is_trampoline_pool_blocked() const {
1361 return trampoline_pool_blocked_nesting_ > 0;
1362 }
1363
1364 bool has_exception() const { return internal_trampoline_exception_; }
1365
1366 bool is_trampoline_emitted() const { return trampoline_emitted_; }
1367
1368 #if V8_OOL_CONSTANT_POOL
1369 void set_constant_pool_available(bool available) {
1370 constant_pool_available_ = available;
1371 }
1372 #endif
1373
1374 private:
1375 // Code generation
1376 // The relocation writer's position is at least kGap bytes below the end of
1377 // the generated instructions. This is so that multi-instruction sequences do
1378 // not have to check for overflow. The same is true for writes of large
1379 // relocation info entries.
1380 static const int kGap = 32;
1381
1382 // Repeated checking whether the trampoline pool should be emitted is rather
1383 // expensive. By default we only check again once a number of instructions
1384 // has been generated.
1385 int next_buffer_check_; // pc offset of next buffer check.
1386
1387 // Emission of the trampoline pool may be blocked in some code sequences.
1388 int trampoline_pool_blocked_nesting_; // Block emission if this is not zero.
1389 int no_trampoline_pool_before_; // Block emission before this pc offset.
1390
1391 // Relocation info generation
1392 // Each relocation is encoded as a variable size value
1393 static const int kMaxRelocSize = RelocInfoWriter::kMaxSize;
1394 RelocInfoWriter reloc_info_writer;
1395
1396 // The bound position, before this we cannot do instruction elimination.
1397 int last_bound_pos_;
1398
1399 #if V8_OOL_CONSTANT_POOL
1400 ConstantPoolBuilder constant_pool_builder_;
1401
1402 // Indicates whether the constant pool can be accessed, which is only possible
1403 // if kConstantPoolRegister points to the current code object's constant pool.
1404 bool constant_pool_available_;
1405 #endif
1406
1407 // Code emission
1408 inline void CheckBuffer();
1409 void GrowBuffer();
1410 inline void emit(Instr x);
1411 inline void CheckTrampolinePoolQuick();
1412
1413 // Instruction generation
1414 void a_form(Instr instr, DoubleRegister frt, DoubleRegister fra,
1415 DoubleRegister frb, RCBit r);
1416 void d_form(Instr instr, Register rt, Register ra, const intptr_t val,
1417 bool signed_disp);
1418 void x_form(Instr instr, Register ra, Register rs, Register rb, RCBit r);
1419 void xo_form(Instr instr, Register rt, Register ra, Register rb, OEBit o,
1420 RCBit r);
1421 void md_form(Instr instr, Register ra, Register rs, int shift, int maskbit,
1422 RCBit r);
1423 void mds_form(Instr instr, Register ra, Register rs, Register rb, int maskbit,
1424 RCBit r);
1425
1426 // Labels
1427 void print(Label* L);
1428 int max_reach_from(int pos);
1429 void bind_to(Label* L, int pos);
1430 void next(Label* L);
1431
1432 class Trampoline {
1433 public:
1434 Trampoline() {
1435 next_slot_ = 0;
1436 free_slot_count_ = 0;
1437 }
1438 Trampoline(int start, int slot_count) {
1439 next_slot_ = start;
1440 free_slot_count_ = slot_count;
1441 }
1442 int take_slot() {
1443 int trampoline_slot = kInvalidSlotPos;
1444 if (free_slot_count_ <= 0) {
1445 // We have run out of space on trampolines.
1446 // Make sure we fail in debug mode, so we become aware of each case
1447 // when this happens.
1448 DCHECK(0);
1449 // Internal exception will be caught.
1450 } else {
1451 trampoline_slot = next_slot_;
1452 free_slot_count_--;
1453 next_slot_ += kTrampolineSlotsSize;
1454 }
1455 return trampoline_slot;
1456 }
1457
1458 private:
1459 int next_slot_;
1460 int free_slot_count_;
1461 };
1462
1463 int32_t get_trampoline_entry();
1464 int unbound_labels_count_;
1465 // If trampoline is emitted, generated code is becoming large. As
1466 // this is already a slow case which can possibly break our code
1467 // generation for the extreme case, we use this information to
1468 // trigger different mode of branch instruction generation, where we
1469 // no longer use a single branch instruction.
1470 bool trampoline_emitted_;
1471 static const int kTrampolineSlotsSize = kInstrSize;
1472 static const int kMaxCondBranchReach = (1 << (16 - 1)) - 1;
1473 static const int kMaxBlockTrampolineSectionSize = 64 * kInstrSize;
1474 static const int kInvalidSlotPos = -1;
1475
1476 Trampoline trampoline_;
1477 bool internal_trampoline_exception_;
1478
1479 friend class RegExpMacroAssemblerPPC;
1480 friend class RelocInfo;
1481 friend class CodePatcher;
1482 friend class BlockTrampolinePoolScope;
1483 #if V8_OOL_CONSTANT_POOL
1484 friend class FrameAndConstantPoolScope;
1485 friend class ConstantPoolUnavailableScope;
1486 #endif
1487
1488 PositionsRecorder positions_recorder_;
1489 friend class PositionsRecorder;
1490 friend class EnsureSpace;
1491 };
1492
1493
1494 class EnsureSpace BASE_EMBEDDED {
1495 public:
1496 explicit EnsureSpace(Assembler* assembler) { assembler->CheckBuffer(); }
1497 };
1498 }
1499 } // namespace v8::internal
1500
1501 #endif // V8_PPC_ASSEMBLER_PPC_H_
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698