blob: fc7ac7061aca14848dd0d4ce8e620f824cb9458a [file] [log] [blame]
jeffhao7fbee072012-08-24 17:56:54 -07001/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "assembler_mips.h"
18
Vladimir Marko80afd022015-05-19 18:08:00 +010019#include "base/bit_utils.h"
Elliott Hughes1aa246d2012-12-13 09:29:36 -080020#include "base/casts.h"
Ian Rogers166db042013-07-26 12:05:57 -070021#include "entrypoints/quick/quick_entrypoints.h"
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020022#include "entrypoints/quick/quick_entrypoints_enum.h"
jeffhao7fbee072012-08-24 17:56:54 -070023#include "memory_region.h"
jeffhao7fbee072012-08-24 17:56:54 -070024#include "thread.h"
25
26namespace art {
27namespace mips {
jeffhao7fbee072012-08-24 17:56:54 -070028
jeffhao7fbee072012-08-24 17:56:54 -070029std::ostream& operator<<(std::ostream& os, const DRegister& rhs) {
30 if (rhs >= D0 && rhs < kNumberOfDRegisters) {
31 os << "d" << static_cast<int>(rhs);
32 } else {
33 os << "DRegister[" << static_cast<int>(rhs) << "]";
34 }
35 return os;
36}
37
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020038void MipsAssembler::FinalizeCode() {
39 for (auto& exception_block : exception_blocks_) {
40 EmitExceptionPoll(&exception_block);
41 }
42 PromoteBranches();
43}
44
45void MipsAssembler::FinalizeInstructions(const MemoryRegion& region) {
Vladimir Marko10ef6942015-10-22 15:25:54 +010046 size_t number_of_delayed_adjust_pcs = cfi().NumberOfDelayedAdvancePCs();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020047 EmitBranches();
48 Assembler::FinalizeInstructions(region);
Vladimir Marko10ef6942015-10-22 15:25:54 +010049 PatchCFI(number_of_delayed_adjust_pcs);
50}
51
52void MipsAssembler::PatchCFI(size_t number_of_delayed_adjust_pcs) {
53 if (cfi().NumberOfDelayedAdvancePCs() == 0u) {
54 DCHECK_EQ(number_of_delayed_adjust_pcs, 0u);
55 return;
56 }
57
58 typedef DebugFrameOpCodeWriterForAssembler::DelayedAdvancePC DelayedAdvancePC;
59 const auto data = cfi().ReleaseStreamAndPrepareForDelayedAdvancePC();
60 const std::vector<uint8_t>& old_stream = data.first;
61 const std::vector<DelayedAdvancePC>& advances = data.second;
62
63 // PCs recorded before EmitBranches() need to be adjusted.
64 // PCs recorded during EmitBranches() are already adjusted.
65 // Both ranges are separately sorted but they may overlap.
66 if (kIsDebugBuild) {
67 auto cmp = [](const DelayedAdvancePC& lhs, const DelayedAdvancePC& rhs) {
68 return lhs.pc < rhs.pc;
69 };
70 CHECK(std::is_sorted(advances.begin(), advances.begin() + number_of_delayed_adjust_pcs, cmp));
71 CHECK(std::is_sorted(advances.begin() + number_of_delayed_adjust_pcs, advances.end(), cmp));
72 }
73
74 // Append initial CFI data if any.
75 size_t size = advances.size();
76 DCHECK_NE(size, 0u);
77 cfi().AppendRawData(old_stream, 0u, advances[0].stream_pos);
78 // Emit PC adjustments interleaved with the old CFI stream.
79 size_t adjust_pos = 0u;
80 size_t late_emit_pos = number_of_delayed_adjust_pcs;
81 while (adjust_pos != number_of_delayed_adjust_pcs || late_emit_pos != size) {
82 size_t adjusted_pc = (adjust_pos != number_of_delayed_adjust_pcs)
83 ? GetAdjustedPosition(advances[adjust_pos].pc)
84 : static_cast<size_t>(-1);
85 size_t late_emit_pc = (late_emit_pos != size)
86 ? advances[late_emit_pos].pc
87 : static_cast<size_t>(-1);
88 size_t advance_pc = std::min(adjusted_pc, late_emit_pc);
89 DCHECK_NE(advance_pc, static_cast<size_t>(-1));
90 size_t entry = (adjusted_pc <= late_emit_pc) ? adjust_pos : late_emit_pos;
91 if (adjusted_pc <= late_emit_pc) {
92 ++adjust_pos;
93 } else {
94 ++late_emit_pos;
95 }
96 cfi().AdvancePC(advance_pc);
97 size_t end_pos = (entry + 1u == size) ? old_stream.size() : advances[entry + 1u].stream_pos;
98 cfi().AppendRawData(old_stream, advances[entry].stream_pos, end_pos);
99 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200100}
101
102void MipsAssembler::EmitBranches() {
103 CHECK(!overwriting_);
104 // Switch from appending instructions at the end of the buffer to overwriting
105 // existing instructions (branch placeholders) in the buffer.
106 overwriting_ = true;
107 for (auto& branch : branches_) {
108 EmitBranch(&branch);
109 }
110 overwriting_ = false;
111}
112
113void MipsAssembler::Emit(uint32_t value) {
114 if (overwriting_) {
115 // Branches to labels are emitted into their placeholders here.
116 buffer_.Store<uint32_t>(overwrite_location_, value);
117 overwrite_location_ += sizeof(uint32_t);
118 } else {
119 // Other instructions are simply appended at the end here.
120 AssemblerBuffer::EnsureCapacity ensured(&buffer_);
121 buffer_.Emit<uint32_t>(value);
122 }
jeffhao7fbee072012-08-24 17:56:54 -0700123}
124
125void MipsAssembler::EmitR(int opcode, Register rs, Register rt, Register rd, int shamt, int funct) {
126 CHECK_NE(rs, kNoRegister);
127 CHECK_NE(rt, kNoRegister);
128 CHECK_NE(rd, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200129 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
130 static_cast<uint32_t>(rs) << kRsShift |
131 static_cast<uint32_t>(rt) << kRtShift |
132 static_cast<uint32_t>(rd) << kRdShift |
133 shamt << kShamtShift |
134 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700135 Emit(encoding);
136}
137
138void MipsAssembler::EmitI(int opcode, Register rs, Register rt, uint16_t imm) {
139 CHECK_NE(rs, kNoRegister);
140 CHECK_NE(rt, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200141 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
142 static_cast<uint32_t>(rs) << kRsShift |
143 static_cast<uint32_t>(rt) << kRtShift |
144 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700145 Emit(encoding);
146}
147
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200148void MipsAssembler::EmitI21(int opcode, Register rs, uint32_t imm21) {
149 CHECK_NE(rs, kNoRegister);
150 CHECK(IsUint<21>(imm21)) << imm21;
151 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
152 static_cast<uint32_t>(rs) << kRsShift |
153 imm21;
jeffhao7fbee072012-08-24 17:56:54 -0700154 Emit(encoding);
155}
156
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200157void MipsAssembler::EmitI26(int opcode, uint32_t imm26) {
158 CHECK(IsUint<26>(imm26)) << imm26;
159 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift | imm26;
160 Emit(encoding);
161}
162
163void MipsAssembler::EmitFR(int opcode, int fmt, FRegister ft, FRegister fs, FRegister fd,
164 int funct) {
jeffhao7fbee072012-08-24 17:56:54 -0700165 CHECK_NE(ft, kNoFRegister);
166 CHECK_NE(fs, kNoFRegister);
167 CHECK_NE(fd, kNoFRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200168 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
169 fmt << kFmtShift |
170 static_cast<uint32_t>(ft) << kFtShift |
171 static_cast<uint32_t>(fs) << kFsShift |
172 static_cast<uint32_t>(fd) << kFdShift |
173 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700174 Emit(encoding);
175}
176
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200177void MipsAssembler::EmitFI(int opcode, int fmt, FRegister ft, uint16_t imm) {
178 CHECK_NE(ft, kNoFRegister);
179 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
180 fmt << kFmtShift |
181 static_cast<uint32_t>(ft) << kFtShift |
182 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700183 Emit(encoding);
184}
185
jeffhao7fbee072012-08-24 17:56:54 -0700186void MipsAssembler::Addu(Register rd, Register rs, Register rt) {
187 EmitR(0, rs, rt, rd, 0, 0x21);
188}
189
jeffhao7fbee072012-08-24 17:56:54 -0700190void MipsAssembler::Addiu(Register rt, Register rs, uint16_t imm16) {
191 EmitI(0x9, rs, rt, imm16);
192}
193
jeffhao7fbee072012-08-24 17:56:54 -0700194void MipsAssembler::Subu(Register rd, Register rs, Register rt) {
195 EmitR(0, rs, rt, rd, 0, 0x23);
196}
197
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200198void MipsAssembler::MultR2(Register rs, Register rt) {
199 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700200 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x18);
201}
202
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200203void MipsAssembler::MultuR2(Register rs, Register rt) {
204 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700205 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x19);
206}
207
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200208void MipsAssembler::DivR2(Register rs, Register rt) {
209 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700210 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1a);
211}
212
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200213void MipsAssembler::DivuR2(Register rs, Register rt) {
214 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700215 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1b);
216}
217
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200218void MipsAssembler::MulR2(Register rd, Register rs, Register rt) {
219 CHECK(!IsR6());
220 EmitR(0x1c, rs, rt, rd, 0, 2);
221}
222
223void MipsAssembler::DivR2(Register rd, Register rs, Register rt) {
224 CHECK(!IsR6());
225 DivR2(rs, rt);
226 Mflo(rd);
227}
228
229void MipsAssembler::ModR2(Register rd, Register rs, Register rt) {
230 CHECK(!IsR6());
231 DivR2(rs, rt);
232 Mfhi(rd);
233}
234
235void MipsAssembler::DivuR2(Register rd, Register rs, Register rt) {
236 CHECK(!IsR6());
237 DivuR2(rs, rt);
238 Mflo(rd);
239}
240
241void MipsAssembler::ModuR2(Register rd, Register rs, Register rt) {
242 CHECK(!IsR6());
243 DivuR2(rs, rt);
244 Mfhi(rd);
245}
246
247void MipsAssembler::MulR6(Register rd, Register rs, Register rt) {
248 CHECK(IsR6());
249 EmitR(0, rs, rt, rd, 2, 0x18);
250}
251
252void MipsAssembler::MuhuR6(Register rd, Register rs, Register rt) {
253 CHECK(IsR6());
254 EmitR(0, rs, rt, rd, 3, 0x19);
255}
256
257void MipsAssembler::DivR6(Register rd, Register rs, Register rt) {
258 CHECK(IsR6());
259 EmitR(0, rs, rt, rd, 2, 0x1a);
260}
261
262void MipsAssembler::ModR6(Register rd, Register rs, Register rt) {
263 CHECK(IsR6());
264 EmitR(0, rs, rt, rd, 3, 0x1a);
265}
266
267void MipsAssembler::DivuR6(Register rd, Register rs, Register rt) {
268 CHECK(IsR6());
269 EmitR(0, rs, rt, rd, 2, 0x1b);
270}
271
272void MipsAssembler::ModuR6(Register rd, Register rs, Register rt) {
273 CHECK(IsR6());
274 EmitR(0, rs, rt, rd, 3, 0x1b);
275}
276
jeffhao7fbee072012-08-24 17:56:54 -0700277void MipsAssembler::And(Register rd, Register rs, Register rt) {
278 EmitR(0, rs, rt, rd, 0, 0x24);
279}
280
281void MipsAssembler::Andi(Register rt, Register rs, uint16_t imm16) {
282 EmitI(0xc, rs, rt, imm16);
283}
284
285void MipsAssembler::Or(Register rd, Register rs, Register rt) {
286 EmitR(0, rs, rt, rd, 0, 0x25);
287}
288
289void MipsAssembler::Ori(Register rt, Register rs, uint16_t imm16) {
290 EmitI(0xd, rs, rt, imm16);
291}
292
293void MipsAssembler::Xor(Register rd, Register rs, Register rt) {
294 EmitR(0, rs, rt, rd, 0, 0x26);
295}
296
297void MipsAssembler::Xori(Register rt, Register rs, uint16_t imm16) {
298 EmitI(0xe, rs, rt, imm16);
299}
300
301void MipsAssembler::Nor(Register rd, Register rs, Register rt) {
302 EmitR(0, rs, rt, rd, 0, 0x27);
303}
304
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200305void MipsAssembler::Seb(Register rd, Register rt) {
306 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x10, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700307}
308
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200309void MipsAssembler::Seh(Register rd, Register rt) {
310 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x18, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700311}
312
Chris Larsen3f8bf652015-10-28 10:08:56 -0700313void MipsAssembler::Wsbh(Register rd, Register rt) {
314 EmitR(0x1f, static_cast<Register>(0), rt, rd, 2, 0x20);
315}
316
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200317void MipsAssembler::Sll(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700318 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200319 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x00);
jeffhao7fbee072012-08-24 17:56:54 -0700320}
321
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200322void MipsAssembler::Srl(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700323 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200324 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x02);
325}
326
Chris Larsen3f8bf652015-10-28 10:08:56 -0700327void MipsAssembler::Rotr(Register rd, Register rt, int shamt) {
328 CHECK(IsUint<5>(shamt)) << shamt;
329 EmitR(0, static_cast<Register>(1), rt, rd, shamt, 0x02);
330}
331
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200332void MipsAssembler::Sra(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700333 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200334 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x03);
335}
336
337void MipsAssembler::Sllv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700338 EmitR(0, rs, rt, rd, 0, 0x04);
339}
340
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200341void MipsAssembler::Srlv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700342 EmitR(0, rs, rt, rd, 0, 0x06);
343}
344
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200345void MipsAssembler::Srav(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700346 EmitR(0, rs, rt, rd, 0, 0x07);
347}
348
349void MipsAssembler::Lb(Register rt, Register rs, uint16_t imm16) {
350 EmitI(0x20, rs, rt, imm16);
351}
352
353void MipsAssembler::Lh(Register rt, Register rs, uint16_t imm16) {
354 EmitI(0x21, rs, rt, imm16);
355}
356
357void MipsAssembler::Lw(Register rt, Register rs, uint16_t imm16) {
358 EmitI(0x23, rs, rt, imm16);
359}
360
361void MipsAssembler::Lbu(Register rt, Register rs, uint16_t imm16) {
362 EmitI(0x24, rs, rt, imm16);
363}
364
365void MipsAssembler::Lhu(Register rt, Register rs, uint16_t imm16) {
366 EmitI(0x25, rs, rt, imm16);
367}
368
369void MipsAssembler::Lui(Register rt, uint16_t imm16) {
370 EmitI(0xf, static_cast<Register>(0), rt, imm16);
371}
372
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200373void MipsAssembler::Sync(uint32_t stype) {
374 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0),
375 stype & 0x1f, 0xf);
376}
377
jeffhao7fbee072012-08-24 17:56:54 -0700378void MipsAssembler::Mfhi(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200379 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700380 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x10);
381}
382
383void MipsAssembler::Mflo(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200384 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700385 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x12);
386}
387
388void MipsAssembler::Sb(Register rt, Register rs, uint16_t imm16) {
389 EmitI(0x28, rs, rt, imm16);
390}
391
392void MipsAssembler::Sh(Register rt, Register rs, uint16_t imm16) {
393 EmitI(0x29, rs, rt, imm16);
394}
395
396void MipsAssembler::Sw(Register rt, Register rs, uint16_t imm16) {
397 EmitI(0x2b, rs, rt, imm16);
398}
399
400void MipsAssembler::Slt(Register rd, Register rs, Register rt) {
401 EmitR(0, rs, rt, rd, 0, 0x2a);
402}
403
404void MipsAssembler::Sltu(Register rd, Register rs, Register rt) {
405 EmitR(0, rs, rt, rd, 0, 0x2b);
406}
407
408void MipsAssembler::Slti(Register rt, Register rs, uint16_t imm16) {
409 EmitI(0xa, rs, rt, imm16);
410}
411
412void MipsAssembler::Sltiu(Register rt, Register rs, uint16_t imm16) {
413 EmitI(0xb, rs, rt, imm16);
414}
415
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200416void MipsAssembler::B(uint16_t imm16) {
417 EmitI(0x4, static_cast<Register>(0), static_cast<Register>(0), imm16);
418}
419
420void MipsAssembler::Beq(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700421 EmitI(0x4, rs, rt, imm16);
422}
423
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200424void MipsAssembler::Bne(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700425 EmitI(0x5, rs, rt, imm16);
426}
427
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200428void MipsAssembler::Beqz(Register rt, uint16_t imm16) {
429 Beq(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700430}
431
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200432void MipsAssembler::Bnez(Register rt, uint16_t imm16) {
433 Bne(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700434}
435
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200436void MipsAssembler::Bltz(Register rt, uint16_t imm16) {
437 EmitI(0x1, rt, static_cast<Register>(0), imm16);
438}
439
440void MipsAssembler::Bgez(Register rt, uint16_t imm16) {
441 EmitI(0x1, rt, static_cast<Register>(0x1), imm16);
442}
443
444void MipsAssembler::Blez(Register rt, uint16_t imm16) {
445 EmitI(0x6, rt, static_cast<Register>(0), imm16);
446}
447
448void MipsAssembler::Bgtz(Register rt, uint16_t imm16) {
449 EmitI(0x7, rt, static_cast<Register>(0), imm16);
450}
451
452void MipsAssembler::J(uint32_t addr26) {
453 EmitI26(0x2, addr26);
454}
455
456void MipsAssembler::Jal(uint32_t addr26) {
457 EmitI26(0x3, addr26);
458}
459
460void MipsAssembler::Jalr(Register rd, Register rs) {
461 EmitR(0, rs, static_cast<Register>(0), rd, 0, 0x09);
jeffhao7fbee072012-08-24 17:56:54 -0700462}
463
464void MipsAssembler::Jalr(Register rs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200465 Jalr(RA, rs);
466}
467
468void MipsAssembler::Jr(Register rs) {
469 Jalr(ZERO, rs);
470}
471
472void MipsAssembler::Nal() {
473 EmitI(0x1, static_cast<Register>(0), static_cast<Register>(0x10), 0);
474}
475
476void MipsAssembler::Auipc(Register rs, uint16_t imm16) {
477 CHECK(IsR6());
478 EmitI(0x3B, rs, static_cast<Register>(0x1E), imm16);
479}
480
481void MipsAssembler::Addiupc(Register rs, uint32_t imm19) {
482 CHECK(IsR6());
483 CHECK(IsUint<19>(imm19)) << imm19;
484 EmitI21(0x3B, rs, imm19);
485}
486
487void MipsAssembler::Bc(uint32_t imm26) {
488 CHECK(IsR6());
489 EmitI26(0x32, imm26);
490}
491
492void MipsAssembler::Jic(Register rt, uint16_t imm16) {
493 CHECK(IsR6());
494 EmitI(0x36, static_cast<Register>(0), rt, imm16);
495}
496
497void MipsAssembler::Jialc(Register rt, uint16_t imm16) {
498 CHECK(IsR6());
499 EmitI(0x3E, static_cast<Register>(0), rt, imm16);
500}
501
502void MipsAssembler::Bltc(Register rs, Register rt, uint16_t imm16) {
503 CHECK(IsR6());
504 CHECK_NE(rs, ZERO);
505 CHECK_NE(rt, ZERO);
506 CHECK_NE(rs, rt);
507 EmitI(0x17, rs, rt, imm16);
508}
509
510void MipsAssembler::Bltzc(Register rt, uint16_t imm16) {
511 CHECK(IsR6());
512 CHECK_NE(rt, ZERO);
513 EmitI(0x17, rt, rt, imm16);
514}
515
516void MipsAssembler::Bgtzc(Register rt, uint16_t imm16) {
517 CHECK(IsR6());
518 CHECK_NE(rt, ZERO);
519 EmitI(0x17, static_cast<Register>(0), rt, imm16);
520}
521
522void MipsAssembler::Bgec(Register rs, Register rt, uint16_t imm16) {
523 CHECK(IsR6());
524 CHECK_NE(rs, ZERO);
525 CHECK_NE(rt, ZERO);
526 CHECK_NE(rs, rt);
527 EmitI(0x16, rs, rt, imm16);
528}
529
530void MipsAssembler::Bgezc(Register rt, uint16_t imm16) {
531 CHECK(IsR6());
532 CHECK_NE(rt, ZERO);
533 EmitI(0x16, rt, rt, imm16);
534}
535
536void MipsAssembler::Blezc(Register rt, uint16_t imm16) {
537 CHECK(IsR6());
538 CHECK_NE(rt, ZERO);
539 EmitI(0x16, static_cast<Register>(0), rt, imm16);
540}
541
542void MipsAssembler::Bltuc(Register rs, Register rt, uint16_t imm16) {
543 CHECK(IsR6());
544 CHECK_NE(rs, ZERO);
545 CHECK_NE(rt, ZERO);
546 CHECK_NE(rs, rt);
547 EmitI(0x7, rs, rt, imm16);
548}
549
550void MipsAssembler::Bgeuc(Register rs, Register rt, uint16_t imm16) {
551 CHECK(IsR6());
552 CHECK_NE(rs, ZERO);
553 CHECK_NE(rt, ZERO);
554 CHECK_NE(rs, rt);
555 EmitI(0x6, rs, rt, imm16);
556}
557
558void MipsAssembler::Beqc(Register rs, Register rt, uint16_t imm16) {
559 CHECK(IsR6());
560 CHECK_NE(rs, ZERO);
561 CHECK_NE(rt, ZERO);
562 CHECK_NE(rs, rt);
563 EmitI(0x8, std::min(rs, rt), std::max(rs, rt), imm16);
564}
565
566void MipsAssembler::Bnec(Register rs, Register rt, uint16_t imm16) {
567 CHECK(IsR6());
568 CHECK_NE(rs, ZERO);
569 CHECK_NE(rt, ZERO);
570 CHECK_NE(rs, rt);
571 EmitI(0x18, std::min(rs, rt), std::max(rs, rt), imm16);
572}
573
574void MipsAssembler::Beqzc(Register rs, uint32_t imm21) {
575 CHECK(IsR6());
576 CHECK_NE(rs, ZERO);
577 EmitI21(0x36, rs, imm21);
578}
579
580void MipsAssembler::Bnezc(Register rs, uint32_t imm21) {
581 CHECK(IsR6());
582 CHECK_NE(rs, ZERO);
583 EmitI21(0x3E, rs, imm21);
584}
585
586void MipsAssembler::EmitBcond(BranchCondition cond, Register rs, Register rt, uint16_t imm16) {
587 switch (cond) {
588 case kCondLTZ:
589 CHECK_EQ(rt, ZERO);
590 Bltz(rs, imm16);
591 break;
592 case kCondGEZ:
593 CHECK_EQ(rt, ZERO);
594 Bgez(rs, imm16);
595 break;
596 case kCondLEZ:
597 CHECK_EQ(rt, ZERO);
598 Blez(rs, imm16);
599 break;
600 case kCondGTZ:
601 CHECK_EQ(rt, ZERO);
602 Bgtz(rs, imm16);
603 break;
604 case kCondEQ:
605 Beq(rs, rt, imm16);
606 break;
607 case kCondNE:
608 Bne(rs, rt, imm16);
609 break;
610 case kCondEQZ:
611 CHECK_EQ(rt, ZERO);
612 Beqz(rs, imm16);
613 break;
614 case kCondNEZ:
615 CHECK_EQ(rt, ZERO);
616 Bnez(rs, imm16);
617 break;
618 case kCondLT:
619 case kCondGE:
620 case kCondLE:
621 case kCondGT:
622 case kCondLTU:
623 case kCondGEU:
624 case kUncond:
625 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
626 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
627 LOG(FATAL) << "Unexpected branch condition " << cond;
628 UNREACHABLE();
629 }
630}
631
632void MipsAssembler::EmitBcondc(BranchCondition cond, Register rs, Register rt, uint32_t imm16_21) {
633 switch (cond) {
634 case kCondLT:
635 Bltc(rs, rt, imm16_21);
636 break;
637 case kCondGE:
638 Bgec(rs, rt, imm16_21);
639 break;
640 case kCondLE:
641 Bgec(rt, rs, imm16_21);
642 break;
643 case kCondGT:
644 Bltc(rt, rs, imm16_21);
645 break;
646 case kCondLTZ:
647 CHECK_EQ(rt, ZERO);
648 Bltzc(rs, imm16_21);
649 break;
650 case kCondGEZ:
651 CHECK_EQ(rt, ZERO);
652 Bgezc(rs, imm16_21);
653 break;
654 case kCondLEZ:
655 CHECK_EQ(rt, ZERO);
656 Blezc(rs, imm16_21);
657 break;
658 case kCondGTZ:
659 CHECK_EQ(rt, ZERO);
660 Bgtzc(rs, imm16_21);
661 break;
662 case kCondEQ:
663 Beqc(rs, rt, imm16_21);
664 break;
665 case kCondNE:
666 Bnec(rs, rt, imm16_21);
667 break;
668 case kCondEQZ:
669 CHECK_EQ(rt, ZERO);
670 Beqzc(rs, imm16_21);
671 break;
672 case kCondNEZ:
673 CHECK_EQ(rt, ZERO);
674 Bnezc(rs, imm16_21);
675 break;
676 case kCondLTU:
677 Bltuc(rs, rt, imm16_21);
678 break;
679 case kCondGEU:
680 Bgeuc(rs, rt, imm16_21);
681 break;
682 case kUncond:
683 LOG(FATAL) << "Unexpected branch condition " << cond;
684 UNREACHABLE();
685 }
jeffhao7fbee072012-08-24 17:56:54 -0700686}
687
688void MipsAssembler::AddS(FRegister fd, FRegister fs, FRegister ft) {
689 EmitFR(0x11, 0x10, ft, fs, fd, 0x0);
690}
691
692void MipsAssembler::SubS(FRegister fd, FRegister fs, FRegister ft) {
693 EmitFR(0x11, 0x10, ft, fs, fd, 0x1);
694}
695
696void MipsAssembler::MulS(FRegister fd, FRegister fs, FRegister ft) {
697 EmitFR(0x11, 0x10, ft, fs, fd, 0x2);
698}
699
700void MipsAssembler::DivS(FRegister fd, FRegister fs, FRegister ft) {
701 EmitFR(0x11, 0x10, ft, fs, fd, 0x3);
702}
703
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200704void MipsAssembler::AddD(FRegister fd, FRegister fs, FRegister ft) {
705 EmitFR(0x11, 0x11, ft, fs, fd, 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700706}
707
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200708void MipsAssembler::SubD(FRegister fd, FRegister fs, FRegister ft) {
709 EmitFR(0x11, 0x11, ft, fs, fd, 0x1);
jeffhao7fbee072012-08-24 17:56:54 -0700710}
711
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200712void MipsAssembler::MulD(FRegister fd, FRegister fs, FRegister ft) {
713 EmitFR(0x11, 0x11, ft, fs, fd, 0x2);
jeffhao7fbee072012-08-24 17:56:54 -0700714}
715
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200716void MipsAssembler::DivD(FRegister fd, FRegister fs, FRegister ft) {
717 EmitFR(0x11, 0x11, ft, fs, fd, 0x3);
jeffhao7fbee072012-08-24 17:56:54 -0700718}
719
720void MipsAssembler::MovS(FRegister fd, FRegister fs) {
721 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x6);
722}
723
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200724void MipsAssembler::MovD(FRegister fd, FRegister fs) {
725 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x6);
726}
727
728void MipsAssembler::NegS(FRegister fd, FRegister fs) {
729 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x7);
730}
731
732void MipsAssembler::NegD(FRegister fd, FRegister fs) {
733 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x7);
734}
735
736void MipsAssembler::Cvtsw(FRegister fd, FRegister fs) {
737 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x20);
738}
739
740void MipsAssembler::Cvtdw(FRegister fd, FRegister fs) {
741 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x21);
742}
743
744void MipsAssembler::Cvtsd(FRegister fd, FRegister fs) {
745 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x20);
746}
747
748void MipsAssembler::Cvtds(FRegister fd, FRegister fs) {
749 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x21);
jeffhao7fbee072012-08-24 17:56:54 -0700750}
751
752void MipsAssembler::Mfc1(Register rt, FRegister fs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200753 EmitFR(0x11, 0x00, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700754}
755
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200756void MipsAssembler::Mtc1(Register rt, FRegister fs) {
757 EmitFR(0x11, 0x04, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
758}
759
760void MipsAssembler::Mfhc1(Register rt, FRegister fs) {
761 EmitFR(0x11, 0x03, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
762}
763
764void MipsAssembler::Mthc1(Register rt, FRegister fs) {
765 EmitFR(0x11, 0x07, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700766}
767
768void MipsAssembler::Lwc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200769 EmitI(0x31, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700770}
771
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200772void MipsAssembler::Ldc1(FRegister ft, Register rs, uint16_t imm16) {
773 EmitI(0x35, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700774}
775
776void MipsAssembler::Swc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200777 EmitI(0x39, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700778}
779
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200780void MipsAssembler::Sdc1(FRegister ft, Register rs, uint16_t imm16) {
781 EmitI(0x3d, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700782}
783
784void MipsAssembler::Break() {
785 EmitR(0, static_cast<Register>(0), static_cast<Register>(0),
786 static_cast<Register>(0), 0, 0xD);
787}
788
jeffhao07030602012-09-26 14:33:14 -0700789void MipsAssembler::Nop() {
790 EmitR(0x0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0), 0, 0x0);
791}
792
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200793void MipsAssembler::Move(Register rd, Register rs) {
794 Or(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700795}
796
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200797void MipsAssembler::Clear(Register rd) {
798 Move(rd, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700799}
800
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200801void MipsAssembler::Not(Register rd, Register rs) {
802 Nor(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700803}
804
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200805void MipsAssembler::Push(Register rs) {
806 IncreaseFrameSize(kMipsWordSize);
807 Sw(rs, SP, 0);
jeffhao7fbee072012-08-24 17:56:54 -0700808}
809
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200810void MipsAssembler::Pop(Register rd) {
811 Lw(rd, SP, 0);
812 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -0700813}
814
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200815void MipsAssembler::PopAndReturn(Register rd, Register rt) {
816 Lw(rd, SP, 0);
817 Jr(rt);
818 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -0700819}
820
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200821void MipsAssembler::LoadConst32(Register rd, int32_t value) {
822 if (IsUint<16>(value)) {
823 // Use OR with (unsigned) immediate to encode 16b unsigned int.
824 Ori(rd, ZERO, value);
825 } else if (IsInt<16>(value)) {
826 // Use ADD with (signed) immediate to encode 16b signed int.
827 Addiu(rd, ZERO, value);
jeffhao7fbee072012-08-24 17:56:54 -0700828 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200829 Lui(rd, High16Bits(value));
830 if (value & 0xFFFF)
831 Ori(rd, rd, Low16Bits(value));
832 }
833}
834
835void MipsAssembler::LoadConst64(Register reg_hi, Register reg_lo, int64_t value) {
836 LoadConst32(reg_lo, Low32Bits(value));
837 LoadConst32(reg_hi, High32Bits(value));
838}
839
840void MipsAssembler::StoreConst32ToOffset(int32_t value,
841 Register base,
842 int32_t offset,
843 Register temp) {
844 if (!IsInt<16>(offset)) {
845 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
846 LoadConst32(AT, offset);
847 Addu(AT, AT, base);
848 base = AT;
849 offset = 0;
850 }
851 LoadConst32(temp, value);
852 Sw(temp, base, offset);
853}
854
855void MipsAssembler::StoreConst64ToOffset(int64_t value,
856 Register base,
857 int32_t offset,
858 Register temp) {
859 // IsInt<16> must be passed a signed value.
860 if (!IsInt<16>(offset) || !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize))) {
861 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
862 LoadConst32(AT, offset);
863 Addu(AT, AT, base);
864 base = AT;
865 offset = 0;
866 }
867 LoadConst32(temp, Low32Bits(value));
868 Sw(temp, base, offset);
869 LoadConst32(temp, High32Bits(value));
870 Sw(temp, base, offset + kMipsWordSize);
871}
872
873void MipsAssembler::LoadSConst32(FRegister r, int32_t value, Register temp) {
874 LoadConst32(temp, value);
875 Mtc1(temp, r);
876}
877
878void MipsAssembler::LoadDConst64(FRegister rd, int64_t value, Register temp) {
879 LoadConst32(temp, Low32Bits(value));
880 Mtc1(temp, rd);
881 LoadConst32(temp, High32Bits(value));
882 Mthc1(temp, rd);
883}
884
885void MipsAssembler::Addiu32(Register rt, Register rs, int32_t value, Register temp) {
886 if (IsInt<16>(value)) {
887 Addiu(rt, rs, value);
888 } else {
889 LoadConst32(temp, value);
890 Addu(rt, rs, temp);
891 }
892}
893
894void MipsAssembler::Branch::InitShortOrLong(MipsAssembler::Branch::OffsetBits offset_size,
895 MipsAssembler::Branch::Type short_type,
896 MipsAssembler::Branch::Type long_type) {
897 type_ = (offset_size <= branch_info_[short_type].offset_size) ? short_type : long_type;
898}
899
900void MipsAssembler::Branch::InitializeType(bool is_call, bool is_r6) {
901 OffsetBits offset_size = GetOffsetSizeNeeded(location_, target_);
902 if (is_r6) {
903 // R6
904 if (is_call) {
905 InitShortOrLong(offset_size, kR6Call, kR6LongCall);
906 } else if (condition_ == kUncond) {
907 InitShortOrLong(offset_size, kR6UncondBranch, kR6LongUncondBranch);
908 } else {
909 if (condition_ == kCondEQZ || condition_ == kCondNEZ) {
910 // Special case for beqzc/bnezc with longer offset than in other b<cond>c instructions.
911 type_ = (offset_size <= kOffset23) ? kR6CondBranch : kR6LongCondBranch;
912 } else {
913 InitShortOrLong(offset_size, kR6CondBranch, kR6LongCondBranch);
914 }
915 }
916 } else {
917 // R2
918 if (is_call) {
919 InitShortOrLong(offset_size, kCall, kLongCall);
920 } else if (condition_ == kUncond) {
921 InitShortOrLong(offset_size, kUncondBranch, kLongUncondBranch);
922 } else {
923 InitShortOrLong(offset_size, kCondBranch, kLongCondBranch);
924 }
925 }
926 old_type_ = type_;
927}
928
929bool MipsAssembler::Branch::IsNop(BranchCondition condition, Register lhs, Register rhs) {
930 switch (condition) {
931 case kCondLT:
932 case kCondGT:
933 case kCondNE:
934 case kCondLTU:
935 return lhs == rhs;
936 default:
937 return false;
938 }
939}
940
941bool MipsAssembler::Branch::IsUncond(BranchCondition condition, Register lhs, Register rhs) {
942 switch (condition) {
943 case kUncond:
944 return true;
945 case kCondGE:
946 case kCondLE:
947 case kCondEQ:
948 case kCondGEU:
949 return lhs == rhs;
950 default:
951 return false;
952 }
953}
954
955MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target)
956 : old_location_(location),
957 location_(location),
958 target_(target),
959 lhs_reg_(0),
960 rhs_reg_(0),
961 condition_(kUncond) {
962 InitializeType(false, is_r6);
963}
964
965MipsAssembler::Branch::Branch(bool is_r6,
966 uint32_t location,
967 uint32_t target,
968 MipsAssembler::BranchCondition condition,
969 Register lhs_reg,
970 Register rhs_reg)
971 : old_location_(location),
972 location_(location),
973 target_(target),
974 lhs_reg_(lhs_reg),
975 rhs_reg_(rhs_reg),
976 condition_(condition) {
977 CHECK_NE(condition, kUncond);
978 switch (condition) {
979 case kCondLT:
980 case kCondGE:
981 case kCondLE:
982 case kCondGT:
983 case kCondLTU:
984 case kCondGEU:
985 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
986 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
987 // We leave this up to the caller.
988 CHECK(is_r6);
989 FALLTHROUGH_INTENDED;
990 case kCondEQ:
991 case kCondNE:
992 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
993 // To compare with 0, use dedicated kCond*Z conditions.
994 CHECK_NE(lhs_reg, ZERO);
995 CHECK_NE(rhs_reg, ZERO);
996 break;
997 case kCondLTZ:
998 case kCondGEZ:
999 case kCondLEZ:
1000 case kCondGTZ:
1001 case kCondEQZ:
1002 case kCondNEZ:
1003 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1004 CHECK_NE(lhs_reg, ZERO);
1005 CHECK_EQ(rhs_reg, ZERO);
1006 break;
1007 case kUncond:
1008 UNREACHABLE();
1009 }
1010 CHECK(!IsNop(condition, lhs_reg, rhs_reg));
1011 if (IsUncond(condition, lhs_reg, rhs_reg)) {
1012 // Branch condition is always true, make the branch unconditional.
1013 condition_ = kUncond;
1014 }
1015 InitializeType(false, is_r6);
1016}
1017
1018MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target, Register indirect_reg)
1019 : old_location_(location),
1020 location_(location),
1021 target_(target),
1022 lhs_reg_(indirect_reg),
1023 rhs_reg_(0),
1024 condition_(kUncond) {
1025 CHECK_NE(indirect_reg, ZERO);
1026 CHECK_NE(indirect_reg, AT);
1027 InitializeType(true, is_r6);
1028}
1029
1030MipsAssembler::BranchCondition MipsAssembler::Branch::OppositeCondition(
1031 MipsAssembler::BranchCondition cond) {
1032 switch (cond) {
1033 case kCondLT:
1034 return kCondGE;
1035 case kCondGE:
1036 return kCondLT;
1037 case kCondLE:
1038 return kCondGT;
1039 case kCondGT:
1040 return kCondLE;
1041 case kCondLTZ:
1042 return kCondGEZ;
1043 case kCondGEZ:
1044 return kCondLTZ;
1045 case kCondLEZ:
1046 return kCondGTZ;
1047 case kCondGTZ:
1048 return kCondLEZ;
1049 case kCondEQ:
1050 return kCondNE;
1051 case kCondNE:
1052 return kCondEQ;
1053 case kCondEQZ:
1054 return kCondNEZ;
1055 case kCondNEZ:
1056 return kCondEQZ;
1057 case kCondLTU:
1058 return kCondGEU;
1059 case kCondGEU:
1060 return kCondLTU;
1061 case kUncond:
1062 LOG(FATAL) << "Unexpected branch condition " << cond;
1063 }
1064 UNREACHABLE();
1065}
1066
1067MipsAssembler::Branch::Type MipsAssembler::Branch::GetType() const {
1068 return type_;
1069}
1070
1071MipsAssembler::BranchCondition MipsAssembler::Branch::GetCondition() const {
1072 return condition_;
1073}
1074
1075Register MipsAssembler::Branch::GetLeftRegister() const {
1076 return static_cast<Register>(lhs_reg_);
1077}
1078
1079Register MipsAssembler::Branch::GetRightRegister() const {
1080 return static_cast<Register>(rhs_reg_);
1081}
1082
1083uint32_t MipsAssembler::Branch::GetTarget() const {
1084 return target_;
1085}
1086
1087uint32_t MipsAssembler::Branch::GetLocation() const {
1088 return location_;
1089}
1090
1091uint32_t MipsAssembler::Branch::GetOldLocation() const {
1092 return old_location_;
1093}
1094
1095uint32_t MipsAssembler::Branch::GetLength() const {
1096 return branch_info_[type_].length;
1097}
1098
1099uint32_t MipsAssembler::Branch::GetOldLength() const {
1100 return branch_info_[old_type_].length;
1101}
1102
1103uint32_t MipsAssembler::Branch::GetSize() const {
1104 return GetLength() * sizeof(uint32_t);
1105}
1106
1107uint32_t MipsAssembler::Branch::GetOldSize() const {
1108 return GetOldLength() * sizeof(uint32_t);
1109}
1110
1111uint32_t MipsAssembler::Branch::GetEndLocation() const {
1112 return GetLocation() + GetSize();
1113}
1114
1115uint32_t MipsAssembler::Branch::GetOldEndLocation() const {
1116 return GetOldLocation() + GetOldSize();
1117}
1118
1119bool MipsAssembler::Branch::IsLong() const {
1120 switch (type_) {
1121 // R2 short branches.
1122 case kUncondBranch:
1123 case kCondBranch:
1124 case kCall:
1125 // R6 short branches.
1126 case kR6UncondBranch:
1127 case kR6CondBranch:
1128 case kR6Call:
1129 return false;
1130 // R2 long branches.
1131 case kLongUncondBranch:
1132 case kLongCondBranch:
1133 case kLongCall:
1134 // R6 long branches.
1135 case kR6LongUncondBranch:
1136 case kR6LongCondBranch:
1137 case kR6LongCall:
1138 return true;
1139 }
1140 UNREACHABLE();
1141}
1142
1143bool MipsAssembler::Branch::IsResolved() const {
1144 return target_ != kUnresolved;
1145}
1146
1147MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSize() const {
1148 OffsetBits offset_size =
1149 (type_ == kR6CondBranch && (condition_ == kCondEQZ || condition_ == kCondNEZ))
1150 ? kOffset23
1151 : branch_info_[type_].offset_size;
1152 return offset_size;
1153}
1154
1155MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSizeNeeded(uint32_t location,
1156 uint32_t target) {
1157 // For unresolved targets assume the shortest encoding
1158 // (later it will be made longer if needed).
1159 if (target == kUnresolved)
1160 return kOffset16;
1161 int64_t distance = static_cast<int64_t>(target) - location;
1162 // To simplify calculations in composite branches consisting of multiple instructions
1163 // bump up the distance by a value larger than the max byte size of a composite branch.
1164 distance += (distance >= 0) ? kMaxBranchSize : -kMaxBranchSize;
1165 if (IsInt<kOffset16>(distance))
1166 return kOffset16;
1167 else if (IsInt<kOffset18>(distance))
1168 return kOffset18;
1169 else if (IsInt<kOffset21>(distance))
1170 return kOffset21;
1171 else if (IsInt<kOffset23>(distance))
1172 return kOffset23;
1173 else if (IsInt<kOffset28>(distance))
1174 return kOffset28;
1175 return kOffset32;
1176}
1177
1178void MipsAssembler::Branch::Resolve(uint32_t target) {
1179 target_ = target;
1180}
1181
1182void MipsAssembler::Branch::Relocate(uint32_t expand_location, uint32_t delta) {
1183 if (location_ > expand_location) {
1184 location_ += delta;
1185 }
1186 if (!IsResolved()) {
1187 return; // Don't know the target yet.
1188 }
1189 if (target_ > expand_location) {
1190 target_ += delta;
1191 }
1192}
1193
1194void MipsAssembler::Branch::PromoteToLong() {
1195 switch (type_) {
1196 // R2 short branches.
1197 case kUncondBranch:
1198 type_ = kLongUncondBranch;
1199 break;
1200 case kCondBranch:
1201 type_ = kLongCondBranch;
1202 break;
1203 case kCall:
1204 type_ = kLongCall;
1205 break;
1206 // R6 short branches.
1207 case kR6UncondBranch:
1208 type_ = kR6LongUncondBranch;
1209 break;
1210 case kR6CondBranch:
1211 type_ = kR6LongCondBranch;
1212 break;
1213 case kR6Call:
1214 type_ = kR6LongCall;
1215 break;
1216 default:
1217 // Note: 'type_' is already long.
1218 break;
1219 }
1220 CHECK(IsLong());
1221}
1222
1223uint32_t MipsAssembler::Branch::PromoteIfNeeded(uint32_t max_short_distance) {
1224 // If the branch is still unresolved or already long, nothing to do.
1225 if (IsLong() || !IsResolved()) {
1226 return 0;
1227 }
1228 // Promote the short branch to long if the offset size is too small
1229 // to hold the distance between location_ and target_.
1230 if (GetOffsetSizeNeeded(location_, target_) > GetOffsetSize()) {
1231 PromoteToLong();
1232 uint32_t old_size = GetOldSize();
1233 uint32_t new_size = GetSize();
1234 CHECK_GT(new_size, old_size);
1235 return new_size - old_size;
1236 }
1237 // The following logic is for debugging/testing purposes.
1238 // Promote some short branches to long when it's not really required.
1239 if (UNLIKELY(max_short_distance != std::numeric_limits<uint32_t>::max())) {
1240 int64_t distance = static_cast<int64_t>(target_) - location_;
1241 distance = (distance >= 0) ? distance : -distance;
1242 if (distance >= max_short_distance) {
1243 PromoteToLong();
1244 uint32_t old_size = GetOldSize();
1245 uint32_t new_size = GetSize();
1246 CHECK_GT(new_size, old_size);
1247 return new_size - old_size;
1248 }
1249 }
1250 return 0;
1251}
1252
1253uint32_t MipsAssembler::Branch::GetOffsetLocation() const {
1254 return location_ + branch_info_[type_].instr_offset * sizeof(uint32_t);
1255}
1256
1257uint32_t MipsAssembler::Branch::GetOffset() const {
1258 CHECK(IsResolved());
1259 uint32_t ofs_mask = 0xFFFFFFFF >> (32 - GetOffsetSize());
1260 // Calculate the byte distance between instructions and also account for
1261 // different PC-relative origins.
1262 uint32_t offset = target_ - GetOffsetLocation() - branch_info_[type_].pc_org * sizeof(uint32_t);
1263 // Prepare the offset for encoding into the instruction(s).
1264 offset = (offset & ofs_mask) >> branch_info_[type_].offset_shift;
1265 return offset;
1266}
1267
1268MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) {
1269 CHECK_LT(branch_id, branches_.size());
1270 return &branches_[branch_id];
1271}
1272
1273const MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) const {
1274 CHECK_LT(branch_id, branches_.size());
1275 return &branches_[branch_id];
1276}
1277
1278void MipsAssembler::Bind(MipsLabel* label) {
1279 CHECK(!label->IsBound());
1280 uint32_t bound_pc = buffer_.Size();
1281
1282 // Walk the list of branches referring to and preceding this label.
1283 // Store the previously unknown target addresses in them.
1284 while (label->IsLinked()) {
1285 uint32_t branch_id = label->Position();
1286 Branch* branch = GetBranch(branch_id);
1287 branch->Resolve(bound_pc);
1288
1289 uint32_t branch_location = branch->GetLocation();
1290 // Extract the location of the previous branch in the list (walking the list backwards;
1291 // the previous branch ID was stored in the space reserved for this branch).
1292 uint32_t prev = buffer_.Load<uint32_t>(branch_location);
1293
1294 // On to the previous branch in the list...
1295 label->position_ = prev;
1296 }
1297
1298 // Now make the label object contain its own location (relative to the end of the preceding
1299 // branch, if any; it will be used by the branches referring to and following this label).
1300 label->prev_branch_id_plus_one_ = branches_.size();
1301 if (label->prev_branch_id_plus_one_) {
1302 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1303 const Branch* branch = GetBranch(branch_id);
1304 bound_pc -= branch->GetEndLocation();
1305 }
1306 label->BindTo(bound_pc);
1307}
1308
1309uint32_t MipsAssembler::GetLabelLocation(MipsLabel* label) const {
1310 CHECK(label->IsBound());
1311 uint32_t target = label->Position();
1312 if (label->prev_branch_id_plus_one_) {
1313 // Get label location based on the branch preceding it.
1314 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1315 const Branch* branch = GetBranch(branch_id);
1316 target += branch->GetEndLocation();
1317 }
1318 return target;
1319}
1320
1321uint32_t MipsAssembler::GetAdjustedPosition(uint32_t old_position) {
1322 // We can reconstruct the adjustment by going through all the branches from the beginning
1323 // up to the old_position. Since we expect AdjustedPosition() to be called in a loop
1324 // with increasing old_position, we can use the data from last AdjustedPosition() to
1325 // continue where we left off and the whole loop should be O(m+n) where m is the number
1326 // of positions to adjust and n is the number of branches.
1327 if (old_position < last_old_position_) {
1328 last_position_adjustment_ = 0;
1329 last_old_position_ = 0;
1330 last_branch_id_ = 0;
1331 }
1332 while (last_branch_id_ != branches_.size()) {
1333 const Branch* branch = GetBranch(last_branch_id_);
1334 if (branch->GetLocation() >= old_position + last_position_adjustment_) {
1335 break;
1336 }
1337 last_position_adjustment_ += branch->GetSize() - branch->GetOldSize();
1338 ++last_branch_id_;
1339 }
1340 last_old_position_ = old_position;
1341 return old_position + last_position_adjustment_;
1342}
1343
1344void MipsAssembler::FinalizeLabeledBranch(MipsLabel* label) {
1345 uint32_t length = branches_.back().GetLength();
1346 if (!label->IsBound()) {
1347 // Branch forward (to a following label), distance is unknown.
1348 // The first branch forward will contain 0, serving as the terminator of
1349 // the list of forward-reaching branches.
1350 Emit(label->position_);
1351 length--;
1352 // Now make the label object point to this branch
1353 // (this forms a linked list of branches preceding this label).
1354 uint32_t branch_id = branches_.size() - 1;
1355 label->LinkTo(branch_id);
1356 }
1357 // Reserve space for the branch.
1358 while (length--) {
1359 Nop();
1360 }
1361}
1362
1363void MipsAssembler::Buncond(MipsLabel* label) {
1364 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1365 branches_.emplace_back(IsR6(), buffer_.Size(), target);
1366 FinalizeLabeledBranch(label);
1367}
1368
1369void MipsAssembler::Bcond(MipsLabel* label, BranchCondition condition, Register lhs, Register rhs) {
1370 // If lhs = rhs, this can be a NOP.
1371 if (Branch::IsNop(condition, lhs, rhs)) {
1372 return;
1373 }
1374 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1375 branches_.emplace_back(IsR6(), buffer_.Size(), target, condition, lhs, rhs);
1376 FinalizeLabeledBranch(label);
1377}
1378
1379void MipsAssembler::Call(MipsLabel* label, Register indirect_reg) {
1380 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1381 branches_.emplace_back(IsR6(), buffer_.Size(), target, indirect_reg);
1382 FinalizeLabeledBranch(label);
1383}
1384
1385void MipsAssembler::PromoteBranches() {
1386 // Promote short branches to long as necessary.
1387 bool changed;
1388 do {
1389 changed = false;
1390 for (auto& branch : branches_) {
1391 CHECK(branch.IsResolved());
1392 uint32_t delta = branch.PromoteIfNeeded();
1393 // If this branch has been promoted and needs to expand in size,
1394 // relocate all branches by the expansion size.
1395 if (delta) {
1396 changed = true;
1397 uint32_t expand_location = branch.GetLocation();
1398 for (auto& branch2 : branches_) {
1399 branch2.Relocate(expand_location, delta);
1400 }
1401 }
1402 }
1403 } while (changed);
1404
1405 // Account for branch expansion by resizing the code buffer
1406 // and moving the code in it to its final location.
1407 size_t branch_count = branches_.size();
1408 if (branch_count > 0) {
1409 // Resize.
1410 Branch& last_branch = branches_[branch_count - 1];
1411 uint32_t size_delta = last_branch.GetEndLocation() - last_branch.GetOldEndLocation();
1412 uint32_t old_size = buffer_.Size();
1413 buffer_.Resize(old_size + size_delta);
1414 // Move the code residing between branch placeholders.
1415 uint32_t end = old_size;
1416 for (size_t i = branch_count; i > 0; ) {
1417 Branch& branch = branches_[--i];
1418 uint32_t size = end - branch.GetOldEndLocation();
1419 buffer_.Move(branch.GetEndLocation(), branch.GetOldEndLocation(), size);
1420 end = branch.GetOldLocation();
1421 }
1422 }
1423}
1424
1425// Note: make sure branch_info_[] and EmitBranch() are kept synchronized.
1426const MipsAssembler::Branch::BranchInfo MipsAssembler::Branch::branch_info_[] = {
1427 // R2 short branches.
1428 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kUncondBranch
1429 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kCondBranch
1430 { 5, 2, 0, MipsAssembler::Branch::kOffset16, 0 }, // kCall
1431 // R2 long branches.
1432 { 9, 3, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongUncondBranch
1433 { 10, 4, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCondBranch
1434 { 6, 1, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCall
1435 // R6 short branches.
1436 { 1, 0, 1, MipsAssembler::Branch::kOffset28, 2 }, // kR6UncondBranch
1437 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kR6CondBranch
1438 // Exception: kOffset23 for beqzc/bnezc.
1439 { 2, 0, 0, MipsAssembler::Branch::kOffset21, 2 }, // kR6Call
1440 // R6 long branches.
1441 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongUncondBranch
1442 { 3, 1, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCondBranch
1443 { 3, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCall
1444};
1445
1446// Note: make sure branch_info_[] and mitBranch() are kept synchronized.
1447void MipsAssembler::EmitBranch(MipsAssembler::Branch* branch) {
1448 CHECK_EQ(overwriting_, true);
1449 overwrite_location_ = branch->GetLocation();
1450 uint32_t offset = branch->GetOffset();
1451 BranchCondition condition = branch->GetCondition();
1452 Register lhs = branch->GetLeftRegister();
1453 Register rhs = branch->GetRightRegister();
1454 switch (branch->GetType()) {
1455 // R2 short branches.
1456 case Branch::kUncondBranch:
1457 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1458 B(offset);
1459 Nop(); // TODO: improve by filling the delay slot.
1460 break;
1461 case Branch::kCondBranch:
1462 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1463 EmitBcond(condition, lhs, rhs, offset);
1464 Nop(); // TODO: improve by filling the delay slot.
1465 break;
1466 case Branch::kCall:
1467 Nal();
1468 Nop(); // TODO: is this NOP really needed here?
1469 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1470 Addiu(lhs, RA, offset);
1471 Jalr(lhs);
1472 Nop();
1473 break;
1474
1475 // R2 long branches.
1476 case Branch::kLongUncondBranch:
1477 // To get the value of the PC register we need to use the NAL instruction.
1478 // NAL clobbers the RA register. However, RA must be preserved if the
1479 // method is compiled without the entry/exit sequences that would take care
1480 // of preserving RA (typically, leaf methods don't preserve RA explicitly).
1481 // So, we need to preserve RA in some temporary storage ourselves. The AT
1482 // register can't be used for this because we need it to load a constant
1483 // which will be added to the value that NAL stores in RA. And we can't
1484 // use T9 for this in the context of the JNI compiler, which uses it
1485 // as a scratch register (see InterproceduralScratchRegister()).
1486 // If we were to add a 32-bit constant to RA using two ADDIU instructions,
1487 // we'd also need to use the ROTR instruction, which requires no less than
1488 // MIPSR2.
1489 // Perhaps, we could use T8 or one of R2's multiplier/divider registers
1490 // (LO or HI) or even a floating-point register, but that doesn't seem
1491 // like a nice solution. We may want this to work on both R6 and pre-R6.
1492 // For now simply use the stack for RA. This should be OK since for the
1493 // vast majority of code a short PC-relative branch is sufficient.
1494 // TODO: can this be improved?
1495 Push(RA);
1496 Nal();
1497 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1498 Lui(AT, High16Bits(offset));
1499 Ori(AT, AT, Low16Bits(offset));
1500 Addu(AT, AT, RA);
1501 Lw(RA, SP, 0);
1502 Jr(AT);
1503 DecreaseFrameSize(kMipsWordSize);
1504 break;
1505 case Branch::kLongCondBranch:
1506 // The comment on case 'Branch::kLongUncondBranch' applies here as well.
1507 // Note: the opposite condition branch encodes 8 as the distance, which is equal to the
1508 // number of instructions skipped:
1509 // (PUSH(IncreaseFrameSize(ADDIU) + SW) + NAL + LUI + ORI + ADDU + LW + JR).
1510 EmitBcond(Branch::OppositeCondition(condition), lhs, rhs, 8);
1511 Push(RA);
1512 Nal();
1513 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1514 Lui(AT, High16Bits(offset));
1515 Ori(AT, AT, Low16Bits(offset));
1516 Addu(AT, AT, RA);
1517 Lw(RA, SP, 0);
1518 Jr(AT);
1519 DecreaseFrameSize(kMipsWordSize);
1520 break;
1521 case Branch::kLongCall:
1522 Nal();
1523 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1524 Lui(AT, High16Bits(offset));
1525 Ori(AT, AT, Low16Bits(offset));
1526 Addu(lhs, AT, RA);
1527 Jalr(lhs);
1528 Nop();
1529 break;
1530
1531 // R6 short branches.
1532 case Branch::kR6UncondBranch:
1533 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1534 Bc(offset);
1535 break;
1536 case Branch::kR6CondBranch:
1537 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1538 EmitBcondc(condition, lhs, rhs, offset);
1539 Nop(); // TODO: improve by filling the forbidden slot.
1540 break;
1541 case Branch::kR6Call:
1542 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1543 Addiupc(lhs, offset);
1544 Jialc(lhs, 0);
1545 break;
1546
1547 // R6 long branches.
1548 case Branch::kR6LongUncondBranch:
1549 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1550 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1551 Auipc(AT, High16Bits(offset));
1552 Jic(AT, Low16Bits(offset));
1553 break;
1554 case Branch::kR6LongCondBranch:
1555 EmitBcondc(Branch::OppositeCondition(condition), lhs, rhs, 2);
1556 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1557 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1558 Auipc(AT, High16Bits(offset));
1559 Jic(AT, Low16Bits(offset));
1560 break;
1561 case Branch::kR6LongCall:
1562 offset += (offset & 0x8000) << 1; // Account for sign extension in addiu.
1563 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1564 Auipc(lhs, High16Bits(offset));
1565 Addiu(lhs, lhs, Low16Bits(offset));
1566 Jialc(lhs, 0);
1567 break;
1568 }
1569 CHECK_EQ(overwrite_location_, branch->GetEndLocation());
1570 CHECK_LT(branch->GetSize(), static_cast<uint32_t>(Branch::kMaxBranchSize));
1571}
1572
1573void MipsAssembler::B(MipsLabel* label) {
1574 Buncond(label);
1575}
1576
1577void MipsAssembler::Jalr(MipsLabel* label, Register indirect_reg) {
1578 Call(label, indirect_reg);
1579}
1580
1581void MipsAssembler::Beq(Register rs, Register rt, MipsLabel* label) {
1582 Bcond(label, kCondEQ, rs, rt);
1583}
1584
1585void MipsAssembler::Bne(Register rs, Register rt, MipsLabel* label) {
1586 Bcond(label, kCondNE, rs, rt);
1587}
1588
1589void MipsAssembler::Beqz(Register rt, MipsLabel* label) {
1590 Bcond(label, kCondEQZ, rt);
1591}
1592
1593void MipsAssembler::Bnez(Register rt, MipsLabel* label) {
1594 Bcond(label, kCondNEZ, rt);
1595}
1596
1597void MipsAssembler::Bltz(Register rt, MipsLabel* label) {
1598 Bcond(label, kCondLTZ, rt);
1599}
1600
1601void MipsAssembler::Bgez(Register rt, MipsLabel* label) {
1602 Bcond(label, kCondGEZ, rt);
1603}
1604
1605void MipsAssembler::Blez(Register rt, MipsLabel* label) {
1606 Bcond(label, kCondLEZ, rt);
1607}
1608
1609void MipsAssembler::Bgtz(Register rt, MipsLabel* label) {
1610 Bcond(label, kCondGTZ, rt);
1611}
1612
1613void MipsAssembler::Blt(Register rs, Register rt, MipsLabel* label) {
1614 if (IsR6()) {
1615 Bcond(label, kCondLT, rs, rt);
1616 } else if (!Branch::IsNop(kCondLT, rs, rt)) {
1617 // Synthesize the instruction (not available on R2).
1618 Slt(AT, rs, rt);
1619 Bnez(AT, label);
1620 }
1621}
1622
1623void MipsAssembler::Bge(Register rs, Register rt, MipsLabel* label) {
1624 if (IsR6()) {
1625 Bcond(label, kCondGE, rs, rt);
1626 } else if (Branch::IsUncond(kCondGE, rs, rt)) {
1627 B(label);
1628 } else {
1629 // Synthesize the instruction (not available on R2).
1630 Slt(AT, rs, rt);
1631 Beqz(AT, label);
1632 }
1633}
1634
1635void MipsAssembler::Bltu(Register rs, Register rt, MipsLabel* label) {
1636 if (IsR6()) {
1637 Bcond(label, kCondLTU, rs, rt);
1638 } else if (!Branch::IsNop(kCondLTU, rs, rt)) {
1639 // Synthesize the instruction (not available on R2).
1640 Sltu(AT, rs, rt);
1641 Bnez(AT, label);
1642 }
1643}
1644
1645void MipsAssembler::Bgeu(Register rs, Register rt, MipsLabel* label) {
1646 if (IsR6()) {
1647 Bcond(label, kCondGEU, rs, rt);
1648 } else if (Branch::IsUncond(kCondGEU, rs, rt)) {
1649 B(label);
1650 } else {
1651 // Synthesize the instruction (not available on R2).
1652 Sltu(AT, rs, rt);
1653 Beqz(AT, label);
jeffhao7fbee072012-08-24 17:56:54 -07001654 }
1655}
1656
1657void MipsAssembler::LoadFromOffset(LoadOperandType type, Register reg, Register base,
1658 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001659 // IsInt<16> must be passed a signed value.
1660 if (!IsInt<16>(offset) ||
1661 (type == kLoadDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1662 LoadConst32(AT, offset);
1663 Addu(AT, AT, base);
1664 base = AT;
1665 offset = 0;
1666 }
1667
jeffhao7fbee072012-08-24 17:56:54 -07001668 switch (type) {
1669 case kLoadSignedByte:
1670 Lb(reg, base, offset);
1671 break;
1672 case kLoadUnsignedByte:
1673 Lbu(reg, base, offset);
1674 break;
1675 case kLoadSignedHalfword:
1676 Lh(reg, base, offset);
1677 break;
1678 case kLoadUnsignedHalfword:
1679 Lhu(reg, base, offset);
1680 break;
1681 case kLoadWord:
1682 Lw(reg, base, offset);
1683 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001684 case kLoadDoubleword:
1685 if (reg == base) {
1686 // This will clobber the base when loading the lower register. Since we have to load the
1687 // higher register as well, this will fail. Solution: reverse the order.
1688 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
1689 Lw(reg, base, offset);
1690 } else {
1691 Lw(reg, base, offset);
1692 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
1693 }
jeffhao7fbee072012-08-24 17:56:54 -07001694 break;
1695 default:
1696 LOG(FATAL) << "UNREACHABLE";
1697 }
1698}
1699
1700void MipsAssembler::LoadSFromOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001701 if (!IsInt<16>(offset)) {
1702 LoadConst32(AT, offset);
1703 Addu(AT, AT, base);
1704 base = AT;
1705 offset = 0;
1706 }
1707
jeffhao7fbee072012-08-24 17:56:54 -07001708 Lwc1(reg, base, offset);
1709}
1710
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001711void MipsAssembler::LoadDFromOffset(FRegister reg, Register base, int32_t offset) {
1712 // IsInt<16> must be passed a signed value.
1713 if (!IsInt<16>(offset) ||
1714 (!IsAligned<kMipsDoublewordSize>(offset) &&
1715 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1716 LoadConst32(AT, offset);
1717 Addu(AT, AT, base);
1718 base = AT;
1719 offset = 0;
1720 }
1721
1722 if (offset & 0x7) {
1723 if (Is32BitFPU()) {
1724 Lwc1(reg, base, offset);
1725 Lwc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
1726 } else {
1727 // 64-bit FPU.
1728 Lwc1(reg, base, offset);
1729 Lw(T8, base, offset + kMipsWordSize);
1730 Mthc1(T8, reg);
1731 }
1732 } else {
1733 Ldc1(reg, base, offset);
1734 }
1735}
1736
1737void MipsAssembler::EmitLoad(ManagedRegister m_dst, Register src_register, int32_t src_offset,
1738 size_t size) {
1739 MipsManagedRegister dst = m_dst.AsMips();
1740 if (dst.IsNoRegister()) {
1741 CHECK_EQ(0u, size) << dst;
1742 } else if (dst.IsCoreRegister()) {
1743 CHECK_EQ(kMipsWordSize, size) << dst;
1744 LoadFromOffset(kLoadWord, dst.AsCoreRegister(), src_register, src_offset);
1745 } else if (dst.IsRegisterPair()) {
1746 CHECK_EQ(kMipsDoublewordSize, size) << dst;
1747 LoadFromOffset(kLoadDoubleword, dst.AsRegisterPairLow(), src_register, src_offset);
1748 } else if (dst.IsFRegister()) {
1749 if (size == kMipsWordSize) {
1750 LoadSFromOffset(dst.AsFRegister(), src_register, src_offset);
1751 } else {
1752 CHECK_EQ(kMipsDoublewordSize, size) << dst;
1753 LoadDFromOffset(dst.AsFRegister(), src_register, src_offset);
1754 }
1755 }
jeffhao7fbee072012-08-24 17:56:54 -07001756}
1757
1758void MipsAssembler::StoreToOffset(StoreOperandType type, Register reg, Register base,
1759 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001760 // IsInt<16> must be passed a signed value.
1761 if (!IsInt<16>(offset) ||
1762 (type == kStoreDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1763 LoadConst32(AT, offset);
1764 Addu(AT, AT, base);
1765 base = AT;
1766 offset = 0;
1767 }
1768
jeffhao7fbee072012-08-24 17:56:54 -07001769 switch (type) {
1770 case kStoreByte:
1771 Sb(reg, base, offset);
1772 break;
1773 case kStoreHalfword:
1774 Sh(reg, base, offset);
1775 break;
1776 case kStoreWord:
1777 Sw(reg, base, offset);
1778 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001779 case kStoreDoubleword:
1780 CHECK_NE(reg, base);
1781 CHECK_NE(static_cast<Register>(reg + 1), base);
1782 Sw(reg, base, offset);
1783 Sw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001784 break;
1785 default:
1786 LOG(FATAL) << "UNREACHABLE";
1787 }
1788}
1789
Goran Jakovljevicff734982015-08-24 12:58:55 +00001790void MipsAssembler::StoreSToOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001791 if (!IsInt<16>(offset)) {
1792 LoadConst32(AT, offset);
1793 Addu(AT, AT, base);
1794 base = AT;
1795 offset = 0;
1796 }
1797
jeffhao7fbee072012-08-24 17:56:54 -07001798 Swc1(reg, base, offset);
1799}
1800
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001801void MipsAssembler::StoreDToOffset(FRegister reg, Register base, int32_t offset) {
1802 // IsInt<16> must be passed a signed value.
1803 if (!IsInt<16>(offset) ||
1804 (!IsAligned<kMipsDoublewordSize>(offset) &&
1805 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1806 LoadConst32(AT, offset);
1807 Addu(AT, AT, base);
1808 base = AT;
1809 offset = 0;
1810 }
1811
1812 if (offset & 0x7) {
1813 if (Is32BitFPU()) {
1814 Swc1(reg, base, offset);
1815 Swc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
1816 } else {
1817 // 64-bit FPU.
1818 Mfhc1(T8, reg);
1819 Swc1(reg, base, offset);
1820 Sw(T8, base, offset + kMipsWordSize);
1821 }
1822 } else {
1823 Sdc1(reg, base, offset);
1824 }
jeffhao7fbee072012-08-24 17:56:54 -07001825}
1826
David Srbeckydd973932015-04-07 20:29:48 +01001827static dwarf::Reg DWARFReg(Register reg) {
1828 return dwarf::Reg::MipsCore(static_cast<int>(reg));
1829}
1830
Ian Rogers790a6b72014-04-01 10:36:00 -07001831constexpr size_t kFramePointerSize = 4;
1832
jeffhao7fbee072012-08-24 17:56:54 -07001833void MipsAssembler::BuildFrame(size_t frame_size, ManagedRegister method_reg,
1834 const std::vector<ManagedRegister>& callee_save_regs,
Dmitry Petrochenkofca82202014-03-21 11:21:37 +07001835 const ManagedRegisterEntrySpills& entry_spills) {
jeffhao7fbee072012-08-24 17:56:54 -07001836 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001837 DCHECK(!overwriting_);
jeffhao7fbee072012-08-24 17:56:54 -07001838
1839 // Increase frame to required size.
1840 IncreaseFrameSize(frame_size);
1841
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001842 // Push callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07001843 int stack_offset = frame_size - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001844 StoreToOffset(kStoreWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001845 cfi_.RelOffset(DWARFReg(RA), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07001846 for (int i = callee_save_regs.size() - 1; i >= 0; --i) {
Ian Rogers790a6b72014-04-01 10:36:00 -07001847 stack_offset -= kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001848 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
1849 StoreToOffset(kStoreWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001850 cfi_.RelOffset(DWARFReg(reg), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07001851 }
1852
1853 // Write out Method*.
1854 StoreToOffset(kStoreWord, method_reg.AsMips().AsCoreRegister(), SP, 0);
1855
1856 // Write out entry spills.
Goran Jakovljevicff734982015-08-24 12:58:55 +00001857 int32_t offset = frame_size + kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001858 for (size_t i = 0; i < entry_spills.size(); ++i) {
Goran Jakovljevicff734982015-08-24 12:58:55 +00001859 MipsManagedRegister reg = entry_spills.at(i).AsMips();
1860 if (reg.IsNoRegister()) {
1861 ManagedRegisterSpill spill = entry_spills.at(i);
1862 offset += spill.getSize();
1863 } else if (reg.IsCoreRegister()) {
1864 StoreToOffset(kStoreWord, reg.AsCoreRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001865 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001866 } else if (reg.IsFRegister()) {
1867 StoreSToOffset(reg.AsFRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001868 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001869 } else if (reg.IsDRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001870 StoreDToOffset(reg.AsOverlappingDRegisterLow(), SP, offset);
1871 offset += kMipsDoublewordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001872 }
jeffhao7fbee072012-08-24 17:56:54 -07001873 }
1874}
1875
1876void MipsAssembler::RemoveFrame(size_t frame_size,
1877 const std::vector<ManagedRegister>& callee_save_regs) {
1878 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001879 DCHECK(!overwriting_);
David Srbeckydd973932015-04-07 20:29:48 +01001880 cfi_.RememberState();
jeffhao7fbee072012-08-24 17:56:54 -07001881
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001882 // Pop callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07001883 int stack_offset = frame_size - (callee_save_regs.size() * kFramePointerSize) - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001884 for (size_t i = 0; i < callee_save_regs.size(); ++i) {
1885 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
1886 LoadFromOffset(kLoadWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001887 cfi_.Restore(DWARFReg(reg));
Ian Rogers790a6b72014-04-01 10:36:00 -07001888 stack_offset += kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001889 }
1890 LoadFromOffset(kLoadWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001891 cfi_.Restore(DWARFReg(RA));
jeffhao7fbee072012-08-24 17:56:54 -07001892
1893 // Decrease frame to required size.
1894 DecreaseFrameSize(frame_size);
jeffhao07030602012-09-26 14:33:14 -07001895
1896 // Then jump to the return address.
1897 Jr(RA);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001898 Nop();
David Srbeckydd973932015-04-07 20:29:48 +01001899
1900 // The CFI should be restored for any code that follows the exit block.
1901 cfi_.RestoreState();
1902 cfi_.DefCFAOffset(frame_size);
jeffhao7fbee072012-08-24 17:56:54 -07001903}
1904
1905void MipsAssembler::IncreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001906 CHECK_ALIGNED(adjust, kFramePointerSize);
1907 Addiu32(SP, SP, -adjust);
David Srbeckydd973932015-04-07 20:29:48 +01001908 cfi_.AdjustCFAOffset(adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001909 if (overwriting_) {
1910 cfi_.OverrideDelayedPC(overwrite_location_);
1911 }
jeffhao7fbee072012-08-24 17:56:54 -07001912}
1913
1914void MipsAssembler::DecreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001915 CHECK_ALIGNED(adjust, kFramePointerSize);
1916 Addiu32(SP, SP, adjust);
David Srbeckydd973932015-04-07 20:29:48 +01001917 cfi_.AdjustCFAOffset(-adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001918 if (overwriting_) {
1919 cfi_.OverrideDelayedPC(overwrite_location_);
1920 }
jeffhao7fbee072012-08-24 17:56:54 -07001921}
1922
1923void MipsAssembler::Store(FrameOffset dest, ManagedRegister msrc, size_t size) {
1924 MipsManagedRegister src = msrc.AsMips();
1925 if (src.IsNoRegister()) {
1926 CHECK_EQ(0u, size);
1927 } else if (src.IsCoreRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001928 CHECK_EQ(kMipsWordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07001929 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
1930 } else if (src.IsRegisterPair()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001931 CHECK_EQ(kMipsDoublewordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07001932 StoreToOffset(kStoreWord, src.AsRegisterPairLow(), SP, dest.Int32Value());
1933 StoreToOffset(kStoreWord, src.AsRegisterPairHigh(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001934 SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001935 } else if (src.IsFRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001936 if (size == kMipsWordSize) {
1937 StoreSToOffset(src.AsFRegister(), SP, dest.Int32Value());
1938 } else {
1939 CHECK_EQ(kMipsDoublewordSize, size);
1940 StoreDToOffset(src.AsFRegister(), SP, dest.Int32Value());
1941 }
jeffhao7fbee072012-08-24 17:56:54 -07001942 }
1943}
1944
1945void MipsAssembler::StoreRef(FrameOffset dest, ManagedRegister msrc) {
1946 MipsManagedRegister src = msrc.AsMips();
1947 CHECK(src.IsCoreRegister());
1948 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
1949}
1950
1951void MipsAssembler::StoreRawPtr(FrameOffset dest, ManagedRegister msrc) {
1952 MipsManagedRegister src = msrc.AsMips();
1953 CHECK(src.IsCoreRegister());
1954 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
1955}
1956
1957void MipsAssembler::StoreImmediateToFrame(FrameOffset dest, uint32_t imm,
1958 ManagedRegister mscratch) {
1959 MipsManagedRegister scratch = mscratch.AsMips();
1960 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001961 LoadConst32(scratch.AsCoreRegister(), imm);
jeffhao7fbee072012-08-24 17:56:54 -07001962 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
1963}
1964
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001965void MipsAssembler::StoreImmediateToThread32(ThreadOffset<kMipsWordSize> dest, uint32_t imm,
jeffhao7fbee072012-08-24 17:56:54 -07001966 ManagedRegister mscratch) {
1967 MipsManagedRegister scratch = mscratch.AsMips();
1968 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001969 // Is this function even referenced anywhere else in the code?
1970 LoadConst32(scratch.AsCoreRegister(), imm);
1971 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), S1, dest.Int32Value());
1972}
1973
1974void MipsAssembler::StoreStackOffsetToThread32(ThreadOffset<kMipsWordSize> thr_offs,
1975 FrameOffset fr_offs,
1976 ManagedRegister mscratch) {
1977 MipsManagedRegister scratch = mscratch.AsMips();
1978 CHECK(scratch.IsCoreRegister()) << scratch;
1979 Addiu32(scratch.AsCoreRegister(), SP, fr_offs.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07001980 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
1981 S1, thr_offs.Int32Value());
1982}
1983
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001984void MipsAssembler::StoreStackPointerToThread32(ThreadOffset<kMipsWordSize> thr_offs) {
jeffhao7fbee072012-08-24 17:56:54 -07001985 StoreToOffset(kStoreWord, SP, S1, thr_offs.Int32Value());
1986}
1987
1988void MipsAssembler::StoreSpanning(FrameOffset dest, ManagedRegister msrc,
1989 FrameOffset in_off, ManagedRegister mscratch) {
1990 MipsManagedRegister src = msrc.AsMips();
1991 MipsManagedRegister scratch = mscratch.AsMips();
1992 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
1993 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, in_off.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001994 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001995}
1996
1997void MipsAssembler::Load(ManagedRegister mdest, FrameOffset src, size_t size) {
1998 return EmitLoad(mdest, SP, src.Int32Value(), size);
1999}
2000
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002001void MipsAssembler::LoadFromThread32(ManagedRegister mdest,
2002 ThreadOffset<kMipsWordSize> src, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002003 return EmitLoad(mdest, S1, src.Int32Value(), size);
2004}
2005
2006void MipsAssembler::LoadRef(ManagedRegister mdest, FrameOffset src) {
2007 MipsManagedRegister dest = mdest.AsMips();
2008 CHECK(dest.IsCoreRegister());
2009 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), SP, src.Int32Value());
2010}
2011
Mathieu Chartiere401d142015-04-22 13:56:20 -07002012void MipsAssembler::LoadRef(ManagedRegister mdest, ManagedRegister base, MemberOffset offs,
Roland Levillain4d027112015-07-01 15:41:14 +01002013 bool unpoison_reference) {
jeffhao7fbee072012-08-24 17:56:54 -07002014 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002015 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002016 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2017 base.AsMips().AsCoreRegister(), offs.Int32Value());
Roland Levillain4d027112015-07-01 15:41:14 +01002018 if (kPoisonHeapReferences && unpoison_reference) {
Hiroshi Yamauchie63a7452014-02-27 14:44:36 -08002019 Subu(dest.AsCoreRegister(), ZERO, dest.AsCoreRegister());
2020 }
jeffhao7fbee072012-08-24 17:56:54 -07002021}
2022
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002023void MipsAssembler::LoadRawPtr(ManagedRegister mdest, ManagedRegister base, Offset offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002024 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002025 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002026 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2027 base.AsMips().AsCoreRegister(), offs.Int32Value());
2028}
2029
Ian Rogersdd7624d2014-03-14 17:43:00 -07002030void MipsAssembler::LoadRawPtrFromThread32(ManagedRegister mdest,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002031 ThreadOffset<kMipsWordSize> offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002032 MipsManagedRegister dest = mdest.AsMips();
2033 CHECK(dest.IsCoreRegister());
2034 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), S1, offs.Int32Value());
2035}
2036
2037void MipsAssembler::SignExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2038 UNIMPLEMENTED(FATAL) << "no sign extension necessary for mips";
2039}
2040
2041void MipsAssembler::ZeroExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2042 UNIMPLEMENTED(FATAL) << "no zero extension necessary for mips";
2043}
2044
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002045void MipsAssembler::Move(ManagedRegister mdest, ManagedRegister msrc, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002046 MipsManagedRegister dest = mdest.AsMips();
2047 MipsManagedRegister src = msrc.AsMips();
2048 if (!dest.Equals(src)) {
2049 if (dest.IsCoreRegister()) {
2050 CHECK(src.IsCoreRegister()) << src;
2051 Move(dest.AsCoreRegister(), src.AsCoreRegister());
2052 } else if (dest.IsFRegister()) {
2053 CHECK(src.IsFRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002054 if (size == kMipsWordSize) {
2055 MovS(dest.AsFRegister(), src.AsFRegister());
2056 } else {
2057 CHECK_EQ(kMipsDoublewordSize, size);
2058 MovD(dest.AsFRegister(), src.AsFRegister());
2059 }
jeffhao7fbee072012-08-24 17:56:54 -07002060 } else if (dest.IsDRegister()) {
2061 CHECK(src.IsDRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002062 MovD(dest.AsOverlappingDRegisterLow(), src.AsOverlappingDRegisterLow());
jeffhao7fbee072012-08-24 17:56:54 -07002063 } else {
2064 CHECK(dest.IsRegisterPair()) << dest;
2065 CHECK(src.IsRegisterPair()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002066 // Ensure that the first move doesn't clobber the input of the second.
jeffhao7fbee072012-08-24 17:56:54 -07002067 if (src.AsRegisterPairHigh() != dest.AsRegisterPairLow()) {
2068 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2069 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2070 } else {
2071 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2072 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2073 }
2074 }
2075 }
2076}
2077
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002078void MipsAssembler::CopyRef(FrameOffset dest, FrameOffset src, ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002079 MipsManagedRegister scratch = mscratch.AsMips();
2080 CHECK(scratch.IsCoreRegister()) << scratch;
2081 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2082 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2083}
2084
Ian Rogersdd7624d2014-03-14 17:43:00 -07002085void MipsAssembler::CopyRawPtrFromThread32(FrameOffset fr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002086 ThreadOffset<kMipsWordSize> thr_offs,
2087 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002088 MipsManagedRegister scratch = mscratch.AsMips();
2089 CHECK(scratch.IsCoreRegister()) << scratch;
2090 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2091 S1, thr_offs.Int32Value());
2092 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2093 SP, fr_offs.Int32Value());
2094}
2095
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002096void MipsAssembler::CopyRawPtrToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2097 FrameOffset fr_offs,
2098 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002099 MipsManagedRegister scratch = mscratch.AsMips();
2100 CHECK(scratch.IsCoreRegister()) << scratch;
2101 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2102 SP, fr_offs.Int32Value());
2103 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2104 S1, thr_offs.Int32Value());
2105}
2106
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002107void MipsAssembler::Copy(FrameOffset dest, FrameOffset src, ManagedRegister mscratch, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002108 MipsManagedRegister scratch = mscratch.AsMips();
2109 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002110 CHECK(size == kMipsWordSize || size == kMipsDoublewordSize) << size;
2111 if (size == kMipsWordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002112 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2113 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002114 } else if (size == kMipsDoublewordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002115 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2116 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002117 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value() + kMipsWordSize);
2118 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002119 }
2120}
2121
2122void MipsAssembler::Copy(FrameOffset dest, ManagedRegister src_base, Offset src_offset,
2123 ManagedRegister mscratch, size_t size) {
2124 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002125 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002126 LoadFromOffset(kLoadWord, scratch, src_base.AsMips().AsCoreRegister(), src_offset.Int32Value());
2127 StoreToOffset(kStoreWord, scratch, SP, dest.Int32Value());
2128}
2129
2130void MipsAssembler::Copy(ManagedRegister dest_base, Offset dest_offset, FrameOffset src,
2131 ManagedRegister mscratch, size_t size) {
2132 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002133 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002134 LoadFromOffset(kLoadWord, scratch, SP, src.Int32Value());
2135 StoreToOffset(kStoreWord, scratch, dest_base.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2136}
2137
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002138void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2139 FrameOffset src_base ATTRIBUTE_UNUSED,
2140 Offset src_offset ATTRIBUTE_UNUSED,
2141 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2142 size_t size ATTRIBUTE_UNUSED) {
2143 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002144}
2145
2146void MipsAssembler::Copy(ManagedRegister dest, Offset dest_offset,
2147 ManagedRegister src, Offset src_offset,
2148 ManagedRegister mscratch, size_t size) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002149 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002150 Register scratch = mscratch.AsMips().AsCoreRegister();
2151 LoadFromOffset(kLoadWord, scratch, src.AsMips().AsCoreRegister(), src_offset.Int32Value());
2152 StoreToOffset(kStoreWord, scratch, dest.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2153}
2154
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002155void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2156 Offset dest_offset ATTRIBUTE_UNUSED,
2157 FrameOffset src ATTRIBUTE_UNUSED,
2158 Offset src_offset ATTRIBUTE_UNUSED,
2159 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2160 size_t size ATTRIBUTE_UNUSED) {
2161 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002162}
2163
2164void MipsAssembler::MemoryBarrier(ManagedRegister) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002165 // TODO: sync?
2166 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002167}
2168
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002169void MipsAssembler::CreateHandleScopeEntry(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002170 FrameOffset handle_scope_offset,
2171 ManagedRegister min_reg,
2172 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002173 MipsManagedRegister out_reg = mout_reg.AsMips();
2174 MipsManagedRegister in_reg = min_reg.AsMips();
2175 CHECK(in_reg.IsNoRegister() || in_reg.IsCoreRegister()) << in_reg;
2176 CHECK(out_reg.IsCoreRegister()) << out_reg;
2177 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002178 MipsLabel null_arg;
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002179 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2180 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002181 // E.g. out_reg = (handle == 0) ? 0 : (SP+handle_offset).
jeffhao7fbee072012-08-24 17:56:54 -07002182 if (in_reg.IsNoRegister()) {
2183 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002184 SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002185 in_reg = out_reg;
2186 }
2187 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002188 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002189 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002190 Beqz(in_reg.AsCoreRegister(), &null_arg);
2191 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2192 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002193 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002194 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002195 }
2196}
2197
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002198void MipsAssembler::CreateHandleScopeEntry(FrameOffset out_off,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002199 FrameOffset handle_scope_offset,
2200 ManagedRegister mscratch,
2201 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002202 MipsManagedRegister scratch = mscratch.AsMips();
2203 CHECK(scratch.IsCoreRegister()) << scratch;
2204 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002205 MipsLabel null_arg;
2206 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002207 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2208 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002209 // E.g. scratch = (scratch == 0) ? 0 : (SP+handle_scope_offset).
2210 Beqz(scratch.AsCoreRegister(), &null_arg);
2211 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2212 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002213 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002214 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002215 }
2216 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, out_off.Int32Value());
2217}
2218
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002219// Given a handle scope entry, load the associated reference.
2220void MipsAssembler::LoadReferenceFromHandleScope(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002221 ManagedRegister min_reg) {
jeffhao7fbee072012-08-24 17:56:54 -07002222 MipsManagedRegister out_reg = mout_reg.AsMips();
2223 MipsManagedRegister in_reg = min_reg.AsMips();
2224 CHECK(out_reg.IsCoreRegister()) << out_reg;
2225 CHECK(in_reg.IsCoreRegister()) << in_reg;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002226 MipsLabel null_arg;
jeffhao7fbee072012-08-24 17:56:54 -07002227 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002228 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002229 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002230 Beqz(in_reg.AsCoreRegister(), &null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002231 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
2232 in_reg.AsCoreRegister(), 0);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002233 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002234}
2235
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002236void MipsAssembler::VerifyObject(ManagedRegister src ATTRIBUTE_UNUSED,
2237 bool could_be_null ATTRIBUTE_UNUSED) {
2238 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002239}
2240
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002241void MipsAssembler::VerifyObject(FrameOffset src ATTRIBUTE_UNUSED,
2242 bool could_be_null ATTRIBUTE_UNUSED) {
2243 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002244}
2245
2246void MipsAssembler::Call(ManagedRegister mbase, Offset offset, ManagedRegister mscratch) {
2247 MipsManagedRegister base = mbase.AsMips();
2248 MipsManagedRegister scratch = mscratch.AsMips();
2249 CHECK(base.IsCoreRegister()) << base;
2250 CHECK(scratch.IsCoreRegister()) << scratch;
2251 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2252 base.AsCoreRegister(), offset.Int32Value());
2253 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002254 Nop();
2255 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002256}
2257
2258void MipsAssembler::Call(FrameOffset base, Offset offset, ManagedRegister mscratch) {
2259 MipsManagedRegister scratch = mscratch.AsMips();
2260 CHECK(scratch.IsCoreRegister()) << scratch;
2261 // Call *(*(SP + base) + offset)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002262 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, base.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002263 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2264 scratch.AsCoreRegister(), offset.Int32Value());
2265 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002266 Nop();
2267 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002268}
2269
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002270void MipsAssembler::CallFromThread32(ThreadOffset<kMipsWordSize> offset ATTRIBUTE_UNUSED,
2271 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
Ian Rogers468532e2013-08-05 10:56:33 -07002272 UNIMPLEMENTED(FATAL) << "no mips implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002273}
2274
2275void MipsAssembler::GetCurrentThread(ManagedRegister tr) {
2276 Move(tr.AsMips().AsCoreRegister(), S1);
2277}
2278
2279void MipsAssembler::GetCurrentThread(FrameOffset offset,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002280 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
jeffhao7fbee072012-08-24 17:56:54 -07002281 StoreToOffset(kStoreWord, S1, SP, offset.Int32Value());
2282}
2283
jeffhao7fbee072012-08-24 17:56:54 -07002284void MipsAssembler::ExceptionPoll(ManagedRegister mscratch, size_t stack_adjust) {
2285 MipsManagedRegister scratch = mscratch.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002286 exception_blocks_.emplace_back(scratch, stack_adjust);
jeffhao7fbee072012-08-24 17:56:54 -07002287 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002288 S1, Thread::ExceptionOffset<kMipsWordSize>().Int32Value());
2289 // TODO: on MIPS32R6 prefer Bnezc(scratch.AsCoreRegister(), slow.Entry());
2290 // as the NAL instruction (occurring in long R2 branches) may become deprecated.
2291 // For now use common for R2 and R6 instructions as this code must execute on both.
2292 Bnez(scratch.AsCoreRegister(), exception_blocks_.back().Entry());
jeffhao7fbee072012-08-24 17:56:54 -07002293}
2294
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002295void MipsAssembler::EmitExceptionPoll(MipsExceptionSlowPath* exception) {
2296 Bind(exception->Entry());
2297 if (exception->stack_adjust_ != 0) { // Fix up the frame.
2298 DecreaseFrameSize(exception->stack_adjust_);
jeffhao7fbee072012-08-24 17:56:54 -07002299 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002300 // Pass exception object as argument.
2301 // Don't care about preserving A0 as this call won't return.
2302 CheckEntrypointTypes<kQuickDeliverException, void, mirror::Object*>();
2303 Move(A0, exception->scratch_.AsCoreRegister());
2304 // Set up call to Thread::Current()->pDeliverException.
2305 LoadFromOffset(kLoadWord, T9, S1,
2306 QUICK_ENTRYPOINT_OFFSET(kMipsWordSize, pDeliverException).Int32Value());
2307 Jr(T9);
2308 Nop();
2309
2310 // Call never returns.
2311 Break();
jeffhao7fbee072012-08-24 17:56:54 -07002312}
2313
2314} // namespace mips
2315} // namespace art