blob: 733ad2cc38d08f641994727c0e6c3e429e769555 [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
Alexey Frunze7e99e052015-11-24 19:28:01 -0800252void MipsAssembler::MuhR6(Register rd, Register rs, Register rt) {
253 CHECK(IsR6());
254 EmitR(0, rs, rt, rd, 3, 0x18);
255}
256
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200257void MipsAssembler::MuhuR6(Register rd, Register rs, Register rt) {
258 CHECK(IsR6());
259 EmitR(0, rs, rt, rd, 3, 0x19);
260}
261
262void MipsAssembler::DivR6(Register rd, Register rs, Register rt) {
263 CHECK(IsR6());
264 EmitR(0, rs, rt, rd, 2, 0x1a);
265}
266
267void MipsAssembler::ModR6(Register rd, Register rs, Register rt) {
268 CHECK(IsR6());
269 EmitR(0, rs, rt, rd, 3, 0x1a);
270}
271
272void MipsAssembler::DivuR6(Register rd, Register rs, Register rt) {
273 CHECK(IsR6());
274 EmitR(0, rs, rt, rd, 2, 0x1b);
275}
276
277void MipsAssembler::ModuR6(Register rd, Register rs, Register rt) {
278 CHECK(IsR6());
279 EmitR(0, rs, rt, rd, 3, 0x1b);
280}
281
jeffhao7fbee072012-08-24 17:56:54 -0700282void MipsAssembler::And(Register rd, Register rs, Register rt) {
283 EmitR(0, rs, rt, rd, 0, 0x24);
284}
285
286void MipsAssembler::Andi(Register rt, Register rs, uint16_t imm16) {
287 EmitI(0xc, rs, rt, imm16);
288}
289
290void MipsAssembler::Or(Register rd, Register rs, Register rt) {
291 EmitR(0, rs, rt, rd, 0, 0x25);
292}
293
294void MipsAssembler::Ori(Register rt, Register rs, uint16_t imm16) {
295 EmitI(0xd, rs, rt, imm16);
296}
297
298void MipsAssembler::Xor(Register rd, Register rs, Register rt) {
299 EmitR(0, rs, rt, rd, 0, 0x26);
300}
301
302void MipsAssembler::Xori(Register rt, Register rs, uint16_t imm16) {
303 EmitI(0xe, rs, rt, imm16);
304}
305
306void MipsAssembler::Nor(Register rd, Register rs, Register rt) {
307 EmitR(0, rs, rt, rd, 0, 0x27);
308}
309
Chris Larsene3845472015-11-18 12:27:15 -0800310void MipsAssembler::Movz(Register rd, Register rs, Register rt) {
311 CHECK(!IsR6());
312 EmitR(0, rs, rt, rd, 0, 0x0A);
313}
314
315void MipsAssembler::Movn(Register rd, Register rs, Register rt) {
316 CHECK(!IsR6());
317 EmitR(0, rs, rt, rd, 0, 0x0B);
318}
319
320void MipsAssembler::Seleqz(Register rd, Register rs, Register rt) {
321 CHECK(IsR6());
322 EmitR(0, rs, rt, rd, 0, 0x35);
323}
324
325void MipsAssembler::Selnez(Register rd, Register rs, Register rt) {
326 CHECK(IsR6());
327 EmitR(0, rs, rt, rd, 0, 0x37);
328}
329
330void MipsAssembler::ClzR6(Register rd, Register rs) {
331 CHECK(IsR6());
332 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x10);
333}
334
335void MipsAssembler::ClzR2(Register rd, Register rs) {
336 CHECK(!IsR6());
337 EmitR(0x1C, rs, rd, rd, 0, 0x20);
338}
339
340void MipsAssembler::CloR6(Register rd, Register rs) {
341 CHECK(IsR6());
342 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x11);
343}
344
345void MipsAssembler::CloR2(Register rd, Register rs) {
346 CHECK(!IsR6());
347 EmitR(0x1C, rs, rd, rd, 0, 0x21);
348}
349
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200350void MipsAssembler::Seb(Register rd, Register rt) {
351 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x10, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700352}
353
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200354void MipsAssembler::Seh(Register rd, Register rt) {
355 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x18, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700356}
357
Chris Larsen3f8bf652015-10-28 10:08:56 -0700358void MipsAssembler::Wsbh(Register rd, Register rt) {
359 EmitR(0x1f, static_cast<Register>(0), rt, rd, 2, 0x20);
360}
361
Chris Larsen70014c82015-11-18 12:26:08 -0800362void MipsAssembler::Bitswap(Register rd, Register rt) {
363 CHECK(IsR6());
364 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x0, 0x20);
365}
366
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200367void MipsAssembler::Sll(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700368 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200369 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x00);
jeffhao7fbee072012-08-24 17:56:54 -0700370}
371
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200372void MipsAssembler::Srl(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700373 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200374 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x02);
375}
376
Chris Larsen3f8bf652015-10-28 10:08:56 -0700377void MipsAssembler::Rotr(Register rd, Register rt, int shamt) {
378 CHECK(IsUint<5>(shamt)) << shamt;
379 EmitR(0, static_cast<Register>(1), rt, rd, shamt, 0x02);
380}
381
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200382void MipsAssembler::Sra(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700383 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200384 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x03);
385}
386
387void MipsAssembler::Sllv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700388 EmitR(0, rs, rt, rd, 0, 0x04);
389}
390
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200391void MipsAssembler::Srlv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700392 EmitR(0, rs, rt, rd, 0, 0x06);
393}
394
Chris Larsene16ce5a2015-11-18 12:30:20 -0800395void MipsAssembler::Rotrv(Register rd, Register rt, Register rs) {
396 EmitR(0, rs, rt, rd, 1, 0x06);
397}
398
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200399void MipsAssembler::Srav(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700400 EmitR(0, rs, rt, rd, 0, 0x07);
401}
402
403void MipsAssembler::Lb(Register rt, Register rs, uint16_t imm16) {
404 EmitI(0x20, rs, rt, imm16);
405}
406
407void MipsAssembler::Lh(Register rt, Register rs, uint16_t imm16) {
408 EmitI(0x21, rs, rt, imm16);
409}
410
411void MipsAssembler::Lw(Register rt, Register rs, uint16_t imm16) {
412 EmitI(0x23, rs, rt, imm16);
413}
414
415void MipsAssembler::Lbu(Register rt, Register rs, uint16_t imm16) {
416 EmitI(0x24, rs, rt, imm16);
417}
418
419void MipsAssembler::Lhu(Register rt, Register rs, uint16_t imm16) {
420 EmitI(0x25, rs, rt, imm16);
421}
422
423void MipsAssembler::Lui(Register rt, uint16_t imm16) {
424 EmitI(0xf, static_cast<Register>(0), rt, imm16);
425}
426
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200427void MipsAssembler::Sync(uint32_t stype) {
428 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0),
429 stype & 0x1f, 0xf);
430}
431
jeffhao7fbee072012-08-24 17:56:54 -0700432void MipsAssembler::Mfhi(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200433 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700434 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x10);
435}
436
437void MipsAssembler::Mflo(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200438 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700439 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x12);
440}
441
442void MipsAssembler::Sb(Register rt, Register rs, uint16_t imm16) {
443 EmitI(0x28, rs, rt, imm16);
444}
445
446void MipsAssembler::Sh(Register rt, Register rs, uint16_t imm16) {
447 EmitI(0x29, rs, rt, imm16);
448}
449
450void MipsAssembler::Sw(Register rt, Register rs, uint16_t imm16) {
451 EmitI(0x2b, rs, rt, imm16);
452}
453
454void MipsAssembler::Slt(Register rd, Register rs, Register rt) {
455 EmitR(0, rs, rt, rd, 0, 0x2a);
456}
457
458void MipsAssembler::Sltu(Register rd, Register rs, Register rt) {
459 EmitR(0, rs, rt, rd, 0, 0x2b);
460}
461
462void MipsAssembler::Slti(Register rt, Register rs, uint16_t imm16) {
463 EmitI(0xa, rs, rt, imm16);
464}
465
466void MipsAssembler::Sltiu(Register rt, Register rs, uint16_t imm16) {
467 EmitI(0xb, rs, rt, imm16);
468}
469
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200470void MipsAssembler::B(uint16_t imm16) {
471 EmitI(0x4, static_cast<Register>(0), static_cast<Register>(0), imm16);
472}
473
474void MipsAssembler::Beq(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700475 EmitI(0x4, rs, rt, imm16);
476}
477
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200478void MipsAssembler::Bne(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700479 EmitI(0x5, rs, rt, imm16);
480}
481
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200482void MipsAssembler::Beqz(Register rt, uint16_t imm16) {
483 Beq(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700484}
485
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200486void MipsAssembler::Bnez(Register rt, uint16_t imm16) {
487 Bne(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700488}
489
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200490void MipsAssembler::Bltz(Register rt, uint16_t imm16) {
491 EmitI(0x1, rt, static_cast<Register>(0), imm16);
492}
493
494void MipsAssembler::Bgez(Register rt, uint16_t imm16) {
495 EmitI(0x1, rt, static_cast<Register>(0x1), imm16);
496}
497
498void MipsAssembler::Blez(Register rt, uint16_t imm16) {
499 EmitI(0x6, rt, static_cast<Register>(0), imm16);
500}
501
502void MipsAssembler::Bgtz(Register rt, uint16_t imm16) {
503 EmitI(0x7, rt, static_cast<Register>(0), imm16);
504}
505
506void MipsAssembler::J(uint32_t addr26) {
507 EmitI26(0x2, addr26);
508}
509
510void MipsAssembler::Jal(uint32_t addr26) {
511 EmitI26(0x3, addr26);
512}
513
514void MipsAssembler::Jalr(Register rd, Register rs) {
515 EmitR(0, rs, static_cast<Register>(0), rd, 0, 0x09);
jeffhao7fbee072012-08-24 17:56:54 -0700516}
517
518void MipsAssembler::Jalr(Register rs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200519 Jalr(RA, rs);
520}
521
522void MipsAssembler::Jr(Register rs) {
523 Jalr(ZERO, rs);
524}
525
526void MipsAssembler::Nal() {
527 EmitI(0x1, static_cast<Register>(0), static_cast<Register>(0x10), 0);
528}
529
530void MipsAssembler::Auipc(Register rs, uint16_t imm16) {
531 CHECK(IsR6());
532 EmitI(0x3B, rs, static_cast<Register>(0x1E), imm16);
533}
534
535void MipsAssembler::Addiupc(Register rs, uint32_t imm19) {
536 CHECK(IsR6());
537 CHECK(IsUint<19>(imm19)) << imm19;
538 EmitI21(0x3B, rs, imm19);
539}
540
541void MipsAssembler::Bc(uint32_t imm26) {
542 CHECK(IsR6());
543 EmitI26(0x32, imm26);
544}
545
546void MipsAssembler::Jic(Register rt, uint16_t imm16) {
547 CHECK(IsR6());
548 EmitI(0x36, static_cast<Register>(0), rt, imm16);
549}
550
551void MipsAssembler::Jialc(Register rt, uint16_t imm16) {
552 CHECK(IsR6());
553 EmitI(0x3E, static_cast<Register>(0), rt, imm16);
554}
555
556void MipsAssembler::Bltc(Register rs, Register rt, uint16_t imm16) {
557 CHECK(IsR6());
558 CHECK_NE(rs, ZERO);
559 CHECK_NE(rt, ZERO);
560 CHECK_NE(rs, rt);
561 EmitI(0x17, rs, rt, imm16);
562}
563
564void MipsAssembler::Bltzc(Register rt, uint16_t imm16) {
565 CHECK(IsR6());
566 CHECK_NE(rt, ZERO);
567 EmitI(0x17, rt, rt, imm16);
568}
569
570void MipsAssembler::Bgtzc(Register rt, uint16_t imm16) {
571 CHECK(IsR6());
572 CHECK_NE(rt, ZERO);
573 EmitI(0x17, static_cast<Register>(0), rt, imm16);
574}
575
576void MipsAssembler::Bgec(Register rs, Register rt, uint16_t imm16) {
577 CHECK(IsR6());
578 CHECK_NE(rs, ZERO);
579 CHECK_NE(rt, ZERO);
580 CHECK_NE(rs, rt);
581 EmitI(0x16, rs, rt, imm16);
582}
583
584void MipsAssembler::Bgezc(Register rt, uint16_t imm16) {
585 CHECK(IsR6());
586 CHECK_NE(rt, ZERO);
587 EmitI(0x16, rt, rt, imm16);
588}
589
590void MipsAssembler::Blezc(Register rt, uint16_t imm16) {
591 CHECK(IsR6());
592 CHECK_NE(rt, ZERO);
593 EmitI(0x16, static_cast<Register>(0), rt, imm16);
594}
595
596void MipsAssembler::Bltuc(Register rs, Register rt, uint16_t imm16) {
597 CHECK(IsR6());
598 CHECK_NE(rs, ZERO);
599 CHECK_NE(rt, ZERO);
600 CHECK_NE(rs, rt);
601 EmitI(0x7, rs, rt, imm16);
602}
603
604void MipsAssembler::Bgeuc(Register rs, Register rt, uint16_t imm16) {
605 CHECK(IsR6());
606 CHECK_NE(rs, ZERO);
607 CHECK_NE(rt, ZERO);
608 CHECK_NE(rs, rt);
609 EmitI(0x6, rs, rt, imm16);
610}
611
612void MipsAssembler::Beqc(Register rs, Register rt, uint16_t imm16) {
613 CHECK(IsR6());
614 CHECK_NE(rs, ZERO);
615 CHECK_NE(rt, ZERO);
616 CHECK_NE(rs, rt);
617 EmitI(0x8, std::min(rs, rt), std::max(rs, rt), imm16);
618}
619
620void MipsAssembler::Bnec(Register rs, Register rt, uint16_t imm16) {
621 CHECK(IsR6());
622 CHECK_NE(rs, ZERO);
623 CHECK_NE(rt, ZERO);
624 CHECK_NE(rs, rt);
625 EmitI(0x18, std::min(rs, rt), std::max(rs, rt), imm16);
626}
627
628void MipsAssembler::Beqzc(Register rs, uint32_t imm21) {
629 CHECK(IsR6());
630 CHECK_NE(rs, ZERO);
631 EmitI21(0x36, rs, imm21);
632}
633
634void MipsAssembler::Bnezc(Register rs, uint32_t imm21) {
635 CHECK(IsR6());
636 CHECK_NE(rs, ZERO);
637 EmitI21(0x3E, rs, imm21);
638}
639
640void MipsAssembler::EmitBcond(BranchCondition cond, Register rs, Register rt, uint16_t imm16) {
641 switch (cond) {
642 case kCondLTZ:
643 CHECK_EQ(rt, ZERO);
644 Bltz(rs, imm16);
645 break;
646 case kCondGEZ:
647 CHECK_EQ(rt, ZERO);
648 Bgez(rs, imm16);
649 break;
650 case kCondLEZ:
651 CHECK_EQ(rt, ZERO);
652 Blez(rs, imm16);
653 break;
654 case kCondGTZ:
655 CHECK_EQ(rt, ZERO);
656 Bgtz(rs, imm16);
657 break;
658 case kCondEQ:
659 Beq(rs, rt, imm16);
660 break;
661 case kCondNE:
662 Bne(rs, rt, imm16);
663 break;
664 case kCondEQZ:
665 CHECK_EQ(rt, ZERO);
666 Beqz(rs, imm16);
667 break;
668 case kCondNEZ:
669 CHECK_EQ(rt, ZERO);
670 Bnez(rs, imm16);
671 break;
672 case kCondLT:
673 case kCondGE:
674 case kCondLE:
675 case kCondGT:
676 case kCondLTU:
677 case kCondGEU:
678 case kUncond:
679 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
680 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
681 LOG(FATAL) << "Unexpected branch condition " << cond;
682 UNREACHABLE();
683 }
684}
685
686void MipsAssembler::EmitBcondc(BranchCondition cond, Register rs, Register rt, uint32_t imm16_21) {
687 switch (cond) {
688 case kCondLT:
689 Bltc(rs, rt, imm16_21);
690 break;
691 case kCondGE:
692 Bgec(rs, rt, imm16_21);
693 break;
694 case kCondLE:
695 Bgec(rt, rs, imm16_21);
696 break;
697 case kCondGT:
698 Bltc(rt, rs, imm16_21);
699 break;
700 case kCondLTZ:
701 CHECK_EQ(rt, ZERO);
702 Bltzc(rs, imm16_21);
703 break;
704 case kCondGEZ:
705 CHECK_EQ(rt, ZERO);
706 Bgezc(rs, imm16_21);
707 break;
708 case kCondLEZ:
709 CHECK_EQ(rt, ZERO);
710 Blezc(rs, imm16_21);
711 break;
712 case kCondGTZ:
713 CHECK_EQ(rt, ZERO);
714 Bgtzc(rs, imm16_21);
715 break;
716 case kCondEQ:
717 Beqc(rs, rt, imm16_21);
718 break;
719 case kCondNE:
720 Bnec(rs, rt, imm16_21);
721 break;
722 case kCondEQZ:
723 CHECK_EQ(rt, ZERO);
724 Beqzc(rs, imm16_21);
725 break;
726 case kCondNEZ:
727 CHECK_EQ(rt, ZERO);
728 Bnezc(rs, imm16_21);
729 break;
730 case kCondLTU:
731 Bltuc(rs, rt, imm16_21);
732 break;
733 case kCondGEU:
734 Bgeuc(rs, rt, imm16_21);
735 break;
736 case kUncond:
737 LOG(FATAL) << "Unexpected branch condition " << cond;
738 UNREACHABLE();
739 }
jeffhao7fbee072012-08-24 17:56:54 -0700740}
741
742void MipsAssembler::AddS(FRegister fd, FRegister fs, FRegister ft) {
743 EmitFR(0x11, 0x10, ft, fs, fd, 0x0);
744}
745
746void MipsAssembler::SubS(FRegister fd, FRegister fs, FRegister ft) {
747 EmitFR(0x11, 0x10, ft, fs, fd, 0x1);
748}
749
750void MipsAssembler::MulS(FRegister fd, FRegister fs, FRegister ft) {
751 EmitFR(0x11, 0x10, ft, fs, fd, 0x2);
752}
753
754void MipsAssembler::DivS(FRegister fd, FRegister fs, FRegister ft) {
755 EmitFR(0x11, 0x10, ft, fs, fd, 0x3);
756}
757
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200758void MipsAssembler::AddD(FRegister fd, FRegister fs, FRegister ft) {
759 EmitFR(0x11, 0x11, ft, fs, fd, 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700760}
761
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200762void MipsAssembler::SubD(FRegister fd, FRegister fs, FRegister ft) {
763 EmitFR(0x11, 0x11, ft, fs, fd, 0x1);
jeffhao7fbee072012-08-24 17:56:54 -0700764}
765
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200766void MipsAssembler::MulD(FRegister fd, FRegister fs, FRegister ft) {
767 EmitFR(0x11, 0x11, ft, fs, fd, 0x2);
jeffhao7fbee072012-08-24 17:56:54 -0700768}
769
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200770void MipsAssembler::DivD(FRegister fd, FRegister fs, FRegister ft) {
771 EmitFR(0x11, 0x11, ft, fs, fd, 0x3);
jeffhao7fbee072012-08-24 17:56:54 -0700772}
773
774void MipsAssembler::MovS(FRegister fd, FRegister fs) {
775 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x6);
776}
777
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200778void MipsAssembler::MovD(FRegister fd, FRegister fs) {
779 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x6);
780}
781
782void MipsAssembler::NegS(FRegister fd, FRegister fs) {
783 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x7);
784}
785
786void MipsAssembler::NegD(FRegister fd, FRegister fs) {
787 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x7);
788}
789
790void MipsAssembler::Cvtsw(FRegister fd, FRegister fs) {
791 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x20);
792}
793
794void MipsAssembler::Cvtdw(FRegister fd, FRegister fs) {
795 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x21);
796}
797
798void MipsAssembler::Cvtsd(FRegister fd, FRegister fs) {
799 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x20);
800}
801
802void MipsAssembler::Cvtds(FRegister fd, FRegister fs) {
803 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x21);
jeffhao7fbee072012-08-24 17:56:54 -0700804}
805
806void MipsAssembler::Mfc1(Register rt, FRegister fs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200807 EmitFR(0x11, 0x00, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700808}
809
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200810void MipsAssembler::Mtc1(Register rt, FRegister fs) {
811 EmitFR(0x11, 0x04, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
812}
813
814void MipsAssembler::Mfhc1(Register rt, FRegister fs) {
815 EmitFR(0x11, 0x03, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
816}
817
818void MipsAssembler::Mthc1(Register rt, FRegister fs) {
819 EmitFR(0x11, 0x07, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700820}
821
822void MipsAssembler::Lwc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200823 EmitI(0x31, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700824}
825
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200826void MipsAssembler::Ldc1(FRegister ft, Register rs, uint16_t imm16) {
827 EmitI(0x35, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700828}
829
830void MipsAssembler::Swc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200831 EmitI(0x39, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700832}
833
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200834void MipsAssembler::Sdc1(FRegister ft, Register rs, uint16_t imm16) {
835 EmitI(0x3d, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700836}
837
838void MipsAssembler::Break() {
839 EmitR(0, static_cast<Register>(0), static_cast<Register>(0),
840 static_cast<Register>(0), 0, 0xD);
841}
842
jeffhao07030602012-09-26 14:33:14 -0700843void MipsAssembler::Nop() {
844 EmitR(0x0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0), 0, 0x0);
845}
846
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200847void MipsAssembler::Move(Register rd, Register rs) {
848 Or(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700849}
850
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200851void MipsAssembler::Clear(Register rd) {
852 Move(rd, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700853}
854
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200855void MipsAssembler::Not(Register rd, Register rs) {
856 Nor(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700857}
858
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200859void MipsAssembler::Push(Register rs) {
860 IncreaseFrameSize(kMipsWordSize);
861 Sw(rs, SP, 0);
jeffhao7fbee072012-08-24 17:56:54 -0700862}
863
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200864void MipsAssembler::Pop(Register rd) {
865 Lw(rd, SP, 0);
866 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -0700867}
868
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200869void MipsAssembler::PopAndReturn(Register rd, Register rt) {
870 Lw(rd, SP, 0);
871 Jr(rt);
872 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -0700873}
874
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200875void MipsAssembler::LoadConst32(Register rd, int32_t value) {
876 if (IsUint<16>(value)) {
877 // Use OR with (unsigned) immediate to encode 16b unsigned int.
878 Ori(rd, ZERO, value);
879 } else if (IsInt<16>(value)) {
880 // Use ADD with (signed) immediate to encode 16b signed int.
881 Addiu(rd, ZERO, value);
jeffhao7fbee072012-08-24 17:56:54 -0700882 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200883 Lui(rd, High16Bits(value));
884 if (value & 0xFFFF)
885 Ori(rd, rd, Low16Bits(value));
886 }
887}
888
889void MipsAssembler::LoadConst64(Register reg_hi, Register reg_lo, int64_t value) {
890 LoadConst32(reg_lo, Low32Bits(value));
891 LoadConst32(reg_hi, High32Bits(value));
892}
893
894void MipsAssembler::StoreConst32ToOffset(int32_t value,
895 Register base,
896 int32_t offset,
897 Register temp) {
898 if (!IsInt<16>(offset)) {
899 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
900 LoadConst32(AT, offset);
901 Addu(AT, AT, base);
902 base = AT;
903 offset = 0;
904 }
905 LoadConst32(temp, value);
906 Sw(temp, base, offset);
907}
908
909void MipsAssembler::StoreConst64ToOffset(int64_t value,
910 Register base,
911 int32_t offset,
912 Register temp) {
913 // IsInt<16> must be passed a signed value.
914 if (!IsInt<16>(offset) || !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize))) {
915 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
916 LoadConst32(AT, offset);
917 Addu(AT, AT, base);
918 base = AT;
919 offset = 0;
920 }
921 LoadConst32(temp, Low32Bits(value));
922 Sw(temp, base, offset);
923 LoadConst32(temp, High32Bits(value));
924 Sw(temp, base, offset + kMipsWordSize);
925}
926
927void MipsAssembler::LoadSConst32(FRegister r, int32_t value, Register temp) {
928 LoadConst32(temp, value);
929 Mtc1(temp, r);
930}
931
932void MipsAssembler::LoadDConst64(FRegister rd, int64_t value, Register temp) {
933 LoadConst32(temp, Low32Bits(value));
934 Mtc1(temp, rd);
935 LoadConst32(temp, High32Bits(value));
936 Mthc1(temp, rd);
937}
938
939void MipsAssembler::Addiu32(Register rt, Register rs, int32_t value, Register temp) {
940 if (IsInt<16>(value)) {
941 Addiu(rt, rs, value);
942 } else {
943 LoadConst32(temp, value);
944 Addu(rt, rs, temp);
945 }
946}
947
948void MipsAssembler::Branch::InitShortOrLong(MipsAssembler::Branch::OffsetBits offset_size,
949 MipsAssembler::Branch::Type short_type,
950 MipsAssembler::Branch::Type long_type) {
951 type_ = (offset_size <= branch_info_[short_type].offset_size) ? short_type : long_type;
952}
953
954void MipsAssembler::Branch::InitializeType(bool is_call, bool is_r6) {
955 OffsetBits offset_size = GetOffsetSizeNeeded(location_, target_);
956 if (is_r6) {
957 // R6
958 if (is_call) {
959 InitShortOrLong(offset_size, kR6Call, kR6LongCall);
960 } else if (condition_ == kUncond) {
961 InitShortOrLong(offset_size, kR6UncondBranch, kR6LongUncondBranch);
962 } else {
963 if (condition_ == kCondEQZ || condition_ == kCondNEZ) {
964 // Special case for beqzc/bnezc with longer offset than in other b<cond>c instructions.
965 type_ = (offset_size <= kOffset23) ? kR6CondBranch : kR6LongCondBranch;
966 } else {
967 InitShortOrLong(offset_size, kR6CondBranch, kR6LongCondBranch);
968 }
969 }
970 } else {
971 // R2
972 if (is_call) {
973 InitShortOrLong(offset_size, kCall, kLongCall);
974 } else if (condition_ == kUncond) {
975 InitShortOrLong(offset_size, kUncondBranch, kLongUncondBranch);
976 } else {
977 InitShortOrLong(offset_size, kCondBranch, kLongCondBranch);
978 }
979 }
980 old_type_ = type_;
981}
982
983bool MipsAssembler::Branch::IsNop(BranchCondition condition, Register lhs, Register rhs) {
984 switch (condition) {
985 case kCondLT:
986 case kCondGT:
987 case kCondNE:
988 case kCondLTU:
989 return lhs == rhs;
990 default:
991 return false;
992 }
993}
994
995bool MipsAssembler::Branch::IsUncond(BranchCondition condition, Register lhs, Register rhs) {
996 switch (condition) {
997 case kUncond:
998 return true;
999 case kCondGE:
1000 case kCondLE:
1001 case kCondEQ:
1002 case kCondGEU:
1003 return lhs == rhs;
1004 default:
1005 return false;
1006 }
1007}
1008
1009MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target)
1010 : old_location_(location),
1011 location_(location),
1012 target_(target),
1013 lhs_reg_(0),
1014 rhs_reg_(0),
1015 condition_(kUncond) {
1016 InitializeType(false, is_r6);
1017}
1018
1019MipsAssembler::Branch::Branch(bool is_r6,
1020 uint32_t location,
1021 uint32_t target,
1022 MipsAssembler::BranchCondition condition,
1023 Register lhs_reg,
1024 Register rhs_reg)
1025 : old_location_(location),
1026 location_(location),
1027 target_(target),
1028 lhs_reg_(lhs_reg),
1029 rhs_reg_(rhs_reg),
1030 condition_(condition) {
1031 CHECK_NE(condition, kUncond);
1032 switch (condition) {
1033 case kCondLT:
1034 case kCondGE:
1035 case kCondLE:
1036 case kCondGT:
1037 case kCondLTU:
1038 case kCondGEU:
1039 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
1040 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
1041 // We leave this up to the caller.
1042 CHECK(is_r6);
1043 FALLTHROUGH_INTENDED;
1044 case kCondEQ:
1045 case kCondNE:
1046 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1047 // To compare with 0, use dedicated kCond*Z conditions.
1048 CHECK_NE(lhs_reg, ZERO);
1049 CHECK_NE(rhs_reg, ZERO);
1050 break;
1051 case kCondLTZ:
1052 case kCondGEZ:
1053 case kCondLEZ:
1054 case kCondGTZ:
1055 case kCondEQZ:
1056 case kCondNEZ:
1057 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1058 CHECK_NE(lhs_reg, ZERO);
1059 CHECK_EQ(rhs_reg, ZERO);
1060 break;
1061 case kUncond:
1062 UNREACHABLE();
1063 }
1064 CHECK(!IsNop(condition, lhs_reg, rhs_reg));
1065 if (IsUncond(condition, lhs_reg, rhs_reg)) {
1066 // Branch condition is always true, make the branch unconditional.
1067 condition_ = kUncond;
1068 }
1069 InitializeType(false, is_r6);
1070}
1071
1072MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target, Register indirect_reg)
1073 : old_location_(location),
1074 location_(location),
1075 target_(target),
1076 lhs_reg_(indirect_reg),
1077 rhs_reg_(0),
1078 condition_(kUncond) {
1079 CHECK_NE(indirect_reg, ZERO);
1080 CHECK_NE(indirect_reg, AT);
1081 InitializeType(true, is_r6);
1082}
1083
1084MipsAssembler::BranchCondition MipsAssembler::Branch::OppositeCondition(
1085 MipsAssembler::BranchCondition cond) {
1086 switch (cond) {
1087 case kCondLT:
1088 return kCondGE;
1089 case kCondGE:
1090 return kCondLT;
1091 case kCondLE:
1092 return kCondGT;
1093 case kCondGT:
1094 return kCondLE;
1095 case kCondLTZ:
1096 return kCondGEZ;
1097 case kCondGEZ:
1098 return kCondLTZ;
1099 case kCondLEZ:
1100 return kCondGTZ;
1101 case kCondGTZ:
1102 return kCondLEZ;
1103 case kCondEQ:
1104 return kCondNE;
1105 case kCondNE:
1106 return kCondEQ;
1107 case kCondEQZ:
1108 return kCondNEZ;
1109 case kCondNEZ:
1110 return kCondEQZ;
1111 case kCondLTU:
1112 return kCondGEU;
1113 case kCondGEU:
1114 return kCondLTU;
1115 case kUncond:
1116 LOG(FATAL) << "Unexpected branch condition " << cond;
1117 }
1118 UNREACHABLE();
1119}
1120
1121MipsAssembler::Branch::Type MipsAssembler::Branch::GetType() const {
1122 return type_;
1123}
1124
1125MipsAssembler::BranchCondition MipsAssembler::Branch::GetCondition() const {
1126 return condition_;
1127}
1128
1129Register MipsAssembler::Branch::GetLeftRegister() const {
1130 return static_cast<Register>(lhs_reg_);
1131}
1132
1133Register MipsAssembler::Branch::GetRightRegister() const {
1134 return static_cast<Register>(rhs_reg_);
1135}
1136
1137uint32_t MipsAssembler::Branch::GetTarget() const {
1138 return target_;
1139}
1140
1141uint32_t MipsAssembler::Branch::GetLocation() const {
1142 return location_;
1143}
1144
1145uint32_t MipsAssembler::Branch::GetOldLocation() const {
1146 return old_location_;
1147}
1148
1149uint32_t MipsAssembler::Branch::GetLength() const {
1150 return branch_info_[type_].length;
1151}
1152
1153uint32_t MipsAssembler::Branch::GetOldLength() const {
1154 return branch_info_[old_type_].length;
1155}
1156
1157uint32_t MipsAssembler::Branch::GetSize() const {
1158 return GetLength() * sizeof(uint32_t);
1159}
1160
1161uint32_t MipsAssembler::Branch::GetOldSize() const {
1162 return GetOldLength() * sizeof(uint32_t);
1163}
1164
1165uint32_t MipsAssembler::Branch::GetEndLocation() const {
1166 return GetLocation() + GetSize();
1167}
1168
1169uint32_t MipsAssembler::Branch::GetOldEndLocation() const {
1170 return GetOldLocation() + GetOldSize();
1171}
1172
1173bool MipsAssembler::Branch::IsLong() const {
1174 switch (type_) {
1175 // R2 short branches.
1176 case kUncondBranch:
1177 case kCondBranch:
1178 case kCall:
1179 // R6 short branches.
1180 case kR6UncondBranch:
1181 case kR6CondBranch:
1182 case kR6Call:
1183 return false;
1184 // R2 long branches.
1185 case kLongUncondBranch:
1186 case kLongCondBranch:
1187 case kLongCall:
1188 // R6 long branches.
1189 case kR6LongUncondBranch:
1190 case kR6LongCondBranch:
1191 case kR6LongCall:
1192 return true;
1193 }
1194 UNREACHABLE();
1195}
1196
1197bool MipsAssembler::Branch::IsResolved() const {
1198 return target_ != kUnresolved;
1199}
1200
1201MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSize() const {
1202 OffsetBits offset_size =
1203 (type_ == kR6CondBranch && (condition_ == kCondEQZ || condition_ == kCondNEZ))
1204 ? kOffset23
1205 : branch_info_[type_].offset_size;
1206 return offset_size;
1207}
1208
1209MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSizeNeeded(uint32_t location,
1210 uint32_t target) {
1211 // For unresolved targets assume the shortest encoding
1212 // (later it will be made longer if needed).
1213 if (target == kUnresolved)
1214 return kOffset16;
1215 int64_t distance = static_cast<int64_t>(target) - location;
1216 // To simplify calculations in composite branches consisting of multiple instructions
1217 // bump up the distance by a value larger than the max byte size of a composite branch.
1218 distance += (distance >= 0) ? kMaxBranchSize : -kMaxBranchSize;
1219 if (IsInt<kOffset16>(distance))
1220 return kOffset16;
1221 else if (IsInt<kOffset18>(distance))
1222 return kOffset18;
1223 else if (IsInt<kOffset21>(distance))
1224 return kOffset21;
1225 else if (IsInt<kOffset23>(distance))
1226 return kOffset23;
1227 else if (IsInt<kOffset28>(distance))
1228 return kOffset28;
1229 return kOffset32;
1230}
1231
1232void MipsAssembler::Branch::Resolve(uint32_t target) {
1233 target_ = target;
1234}
1235
1236void MipsAssembler::Branch::Relocate(uint32_t expand_location, uint32_t delta) {
1237 if (location_ > expand_location) {
1238 location_ += delta;
1239 }
1240 if (!IsResolved()) {
1241 return; // Don't know the target yet.
1242 }
1243 if (target_ > expand_location) {
1244 target_ += delta;
1245 }
1246}
1247
1248void MipsAssembler::Branch::PromoteToLong() {
1249 switch (type_) {
1250 // R2 short branches.
1251 case kUncondBranch:
1252 type_ = kLongUncondBranch;
1253 break;
1254 case kCondBranch:
1255 type_ = kLongCondBranch;
1256 break;
1257 case kCall:
1258 type_ = kLongCall;
1259 break;
1260 // R6 short branches.
1261 case kR6UncondBranch:
1262 type_ = kR6LongUncondBranch;
1263 break;
1264 case kR6CondBranch:
1265 type_ = kR6LongCondBranch;
1266 break;
1267 case kR6Call:
1268 type_ = kR6LongCall;
1269 break;
1270 default:
1271 // Note: 'type_' is already long.
1272 break;
1273 }
1274 CHECK(IsLong());
1275}
1276
1277uint32_t MipsAssembler::Branch::PromoteIfNeeded(uint32_t max_short_distance) {
1278 // If the branch is still unresolved or already long, nothing to do.
1279 if (IsLong() || !IsResolved()) {
1280 return 0;
1281 }
1282 // Promote the short branch to long if the offset size is too small
1283 // to hold the distance between location_ and target_.
1284 if (GetOffsetSizeNeeded(location_, target_) > GetOffsetSize()) {
1285 PromoteToLong();
1286 uint32_t old_size = GetOldSize();
1287 uint32_t new_size = GetSize();
1288 CHECK_GT(new_size, old_size);
1289 return new_size - old_size;
1290 }
1291 // The following logic is for debugging/testing purposes.
1292 // Promote some short branches to long when it's not really required.
1293 if (UNLIKELY(max_short_distance != std::numeric_limits<uint32_t>::max())) {
1294 int64_t distance = static_cast<int64_t>(target_) - location_;
1295 distance = (distance >= 0) ? distance : -distance;
1296 if (distance >= max_short_distance) {
1297 PromoteToLong();
1298 uint32_t old_size = GetOldSize();
1299 uint32_t new_size = GetSize();
1300 CHECK_GT(new_size, old_size);
1301 return new_size - old_size;
1302 }
1303 }
1304 return 0;
1305}
1306
1307uint32_t MipsAssembler::Branch::GetOffsetLocation() const {
1308 return location_ + branch_info_[type_].instr_offset * sizeof(uint32_t);
1309}
1310
1311uint32_t MipsAssembler::Branch::GetOffset() const {
1312 CHECK(IsResolved());
1313 uint32_t ofs_mask = 0xFFFFFFFF >> (32 - GetOffsetSize());
1314 // Calculate the byte distance between instructions and also account for
1315 // different PC-relative origins.
1316 uint32_t offset = target_ - GetOffsetLocation() - branch_info_[type_].pc_org * sizeof(uint32_t);
1317 // Prepare the offset for encoding into the instruction(s).
1318 offset = (offset & ofs_mask) >> branch_info_[type_].offset_shift;
1319 return offset;
1320}
1321
1322MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) {
1323 CHECK_LT(branch_id, branches_.size());
1324 return &branches_[branch_id];
1325}
1326
1327const MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) const {
1328 CHECK_LT(branch_id, branches_.size());
1329 return &branches_[branch_id];
1330}
1331
1332void MipsAssembler::Bind(MipsLabel* label) {
1333 CHECK(!label->IsBound());
1334 uint32_t bound_pc = buffer_.Size();
1335
1336 // Walk the list of branches referring to and preceding this label.
1337 // Store the previously unknown target addresses in them.
1338 while (label->IsLinked()) {
1339 uint32_t branch_id = label->Position();
1340 Branch* branch = GetBranch(branch_id);
1341 branch->Resolve(bound_pc);
1342
1343 uint32_t branch_location = branch->GetLocation();
1344 // Extract the location of the previous branch in the list (walking the list backwards;
1345 // the previous branch ID was stored in the space reserved for this branch).
1346 uint32_t prev = buffer_.Load<uint32_t>(branch_location);
1347
1348 // On to the previous branch in the list...
1349 label->position_ = prev;
1350 }
1351
1352 // Now make the label object contain its own location (relative to the end of the preceding
1353 // branch, if any; it will be used by the branches referring to and following this label).
1354 label->prev_branch_id_plus_one_ = branches_.size();
1355 if (label->prev_branch_id_plus_one_) {
1356 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1357 const Branch* branch = GetBranch(branch_id);
1358 bound_pc -= branch->GetEndLocation();
1359 }
1360 label->BindTo(bound_pc);
1361}
1362
1363uint32_t MipsAssembler::GetLabelLocation(MipsLabel* label) const {
1364 CHECK(label->IsBound());
1365 uint32_t target = label->Position();
1366 if (label->prev_branch_id_plus_one_) {
1367 // Get label location based on the branch preceding it.
1368 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1369 const Branch* branch = GetBranch(branch_id);
1370 target += branch->GetEndLocation();
1371 }
1372 return target;
1373}
1374
1375uint32_t MipsAssembler::GetAdjustedPosition(uint32_t old_position) {
1376 // We can reconstruct the adjustment by going through all the branches from the beginning
1377 // up to the old_position. Since we expect AdjustedPosition() to be called in a loop
1378 // with increasing old_position, we can use the data from last AdjustedPosition() to
1379 // continue where we left off and the whole loop should be O(m+n) where m is the number
1380 // of positions to adjust and n is the number of branches.
1381 if (old_position < last_old_position_) {
1382 last_position_adjustment_ = 0;
1383 last_old_position_ = 0;
1384 last_branch_id_ = 0;
1385 }
1386 while (last_branch_id_ != branches_.size()) {
1387 const Branch* branch = GetBranch(last_branch_id_);
1388 if (branch->GetLocation() >= old_position + last_position_adjustment_) {
1389 break;
1390 }
1391 last_position_adjustment_ += branch->GetSize() - branch->GetOldSize();
1392 ++last_branch_id_;
1393 }
1394 last_old_position_ = old_position;
1395 return old_position + last_position_adjustment_;
1396}
1397
1398void MipsAssembler::FinalizeLabeledBranch(MipsLabel* label) {
1399 uint32_t length = branches_.back().GetLength();
1400 if (!label->IsBound()) {
1401 // Branch forward (to a following label), distance is unknown.
1402 // The first branch forward will contain 0, serving as the terminator of
1403 // the list of forward-reaching branches.
1404 Emit(label->position_);
1405 length--;
1406 // Now make the label object point to this branch
1407 // (this forms a linked list of branches preceding this label).
1408 uint32_t branch_id = branches_.size() - 1;
1409 label->LinkTo(branch_id);
1410 }
1411 // Reserve space for the branch.
1412 while (length--) {
1413 Nop();
1414 }
1415}
1416
1417void MipsAssembler::Buncond(MipsLabel* label) {
1418 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1419 branches_.emplace_back(IsR6(), buffer_.Size(), target);
1420 FinalizeLabeledBranch(label);
1421}
1422
1423void MipsAssembler::Bcond(MipsLabel* label, BranchCondition condition, Register lhs, Register rhs) {
1424 // If lhs = rhs, this can be a NOP.
1425 if (Branch::IsNop(condition, lhs, rhs)) {
1426 return;
1427 }
1428 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1429 branches_.emplace_back(IsR6(), buffer_.Size(), target, condition, lhs, rhs);
1430 FinalizeLabeledBranch(label);
1431}
1432
1433void MipsAssembler::Call(MipsLabel* label, Register indirect_reg) {
1434 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1435 branches_.emplace_back(IsR6(), buffer_.Size(), target, indirect_reg);
1436 FinalizeLabeledBranch(label);
1437}
1438
1439void MipsAssembler::PromoteBranches() {
1440 // Promote short branches to long as necessary.
1441 bool changed;
1442 do {
1443 changed = false;
1444 for (auto& branch : branches_) {
1445 CHECK(branch.IsResolved());
1446 uint32_t delta = branch.PromoteIfNeeded();
1447 // If this branch has been promoted and needs to expand in size,
1448 // relocate all branches by the expansion size.
1449 if (delta) {
1450 changed = true;
1451 uint32_t expand_location = branch.GetLocation();
1452 for (auto& branch2 : branches_) {
1453 branch2.Relocate(expand_location, delta);
1454 }
1455 }
1456 }
1457 } while (changed);
1458
1459 // Account for branch expansion by resizing the code buffer
1460 // and moving the code in it to its final location.
1461 size_t branch_count = branches_.size();
1462 if (branch_count > 0) {
1463 // Resize.
1464 Branch& last_branch = branches_[branch_count - 1];
1465 uint32_t size_delta = last_branch.GetEndLocation() - last_branch.GetOldEndLocation();
1466 uint32_t old_size = buffer_.Size();
1467 buffer_.Resize(old_size + size_delta);
1468 // Move the code residing between branch placeholders.
1469 uint32_t end = old_size;
1470 for (size_t i = branch_count; i > 0; ) {
1471 Branch& branch = branches_[--i];
1472 uint32_t size = end - branch.GetOldEndLocation();
1473 buffer_.Move(branch.GetEndLocation(), branch.GetOldEndLocation(), size);
1474 end = branch.GetOldLocation();
1475 }
1476 }
1477}
1478
1479// Note: make sure branch_info_[] and EmitBranch() are kept synchronized.
1480const MipsAssembler::Branch::BranchInfo MipsAssembler::Branch::branch_info_[] = {
1481 // R2 short branches.
1482 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kUncondBranch
1483 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kCondBranch
1484 { 5, 2, 0, MipsAssembler::Branch::kOffset16, 0 }, // kCall
1485 // R2 long branches.
1486 { 9, 3, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongUncondBranch
1487 { 10, 4, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCondBranch
1488 { 6, 1, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCall
1489 // R6 short branches.
1490 { 1, 0, 1, MipsAssembler::Branch::kOffset28, 2 }, // kR6UncondBranch
1491 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kR6CondBranch
1492 // Exception: kOffset23 for beqzc/bnezc.
1493 { 2, 0, 0, MipsAssembler::Branch::kOffset21, 2 }, // kR6Call
1494 // R6 long branches.
1495 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongUncondBranch
1496 { 3, 1, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCondBranch
1497 { 3, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCall
1498};
1499
1500// Note: make sure branch_info_[] and mitBranch() are kept synchronized.
1501void MipsAssembler::EmitBranch(MipsAssembler::Branch* branch) {
1502 CHECK_EQ(overwriting_, true);
1503 overwrite_location_ = branch->GetLocation();
1504 uint32_t offset = branch->GetOffset();
1505 BranchCondition condition = branch->GetCondition();
1506 Register lhs = branch->GetLeftRegister();
1507 Register rhs = branch->GetRightRegister();
1508 switch (branch->GetType()) {
1509 // R2 short branches.
1510 case Branch::kUncondBranch:
1511 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1512 B(offset);
1513 Nop(); // TODO: improve by filling the delay slot.
1514 break;
1515 case Branch::kCondBranch:
1516 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1517 EmitBcond(condition, lhs, rhs, offset);
1518 Nop(); // TODO: improve by filling the delay slot.
1519 break;
1520 case Branch::kCall:
1521 Nal();
1522 Nop(); // TODO: is this NOP really needed here?
1523 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1524 Addiu(lhs, RA, offset);
1525 Jalr(lhs);
1526 Nop();
1527 break;
1528
1529 // R2 long branches.
1530 case Branch::kLongUncondBranch:
1531 // To get the value of the PC register we need to use the NAL instruction.
1532 // NAL clobbers the RA register. However, RA must be preserved if the
1533 // method is compiled without the entry/exit sequences that would take care
1534 // of preserving RA (typically, leaf methods don't preserve RA explicitly).
1535 // So, we need to preserve RA in some temporary storage ourselves. The AT
1536 // register can't be used for this because we need it to load a constant
1537 // which will be added to the value that NAL stores in RA. And we can't
1538 // use T9 for this in the context of the JNI compiler, which uses it
1539 // as a scratch register (see InterproceduralScratchRegister()).
1540 // If we were to add a 32-bit constant to RA using two ADDIU instructions,
1541 // we'd also need to use the ROTR instruction, which requires no less than
1542 // MIPSR2.
1543 // Perhaps, we could use T8 or one of R2's multiplier/divider registers
1544 // (LO or HI) or even a floating-point register, but that doesn't seem
1545 // like a nice solution. We may want this to work on both R6 and pre-R6.
1546 // For now simply use the stack for RA. This should be OK since for the
1547 // vast majority of code a short PC-relative branch is sufficient.
1548 // TODO: can this be improved?
1549 Push(RA);
1550 Nal();
1551 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1552 Lui(AT, High16Bits(offset));
1553 Ori(AT, AT, Low16Bits(offset));
1554 Addu(AT, AT, RA);
1555 Lw(RA, SP, 0);
1556 Jr(AT);
1557 DecreaseFrameSize(kMipsWordSize);
1558 break;
1559 case Branch::kLongCondBranch:
1560 // The comment on case 'Branch::kLongUncondBranch' applies here as well.
1561 // Note: the opposite condition branch encodes 8 as the distance, which is equal to the
1562 // number of instructions skipped:
1563 // (PUSH(IncreaseFrameSize(ADDIU) + SW) + NAL + LUI + ORI + ADDU + LW + JR).
1564 EmitBcond(Branch::OppositeCondition(condition), lhs, rhs, 8);
1565 Push(RA);
1566 Nal();
1567 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1568 Lui(AT, High16Bits(offset));
1569 Ori(AT, AT, Low16Bits(offset));
1570 Addu(AT, AT, RA);
1571 Lw(RA, SP, 0);
1572 Jr(AT);
1573 DecreaseFrameSize(kMipsWordSize);
1574 break;
1575 case Branch::kLongCall:
1576 Nal();
1577 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1578 Lui(AT, High16Bits(offset));
1579 Ori(AT, AT, Low16Bits(offset));
1580 Addu(lhs, AT, RA);
1581 Jalr(lhs);
1582 Nop();
1583 break;
1584
1585 // R6 short branches.
1586 case Branch::kR6UncondBranch:
1587 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1588 Bc(offset);
1589 break;
1590 case Branch::kR6CondBranch:
1591 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1592 EmitBcondc(condition, lhs, rhs, offset);
1593 Nop(); // TODO: improve by filling the forbidden slot.
1594 break;
1595 case Branch::kR6Call:
1596 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1597 Addiupc(lhs, offset);
1598 Jialc(lhs, 0);
1599 break;
1600
1601 // R6 long branches.
1602 case Branch::kR6LongUncondBranch:
1603 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1604 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1605 Auipc(AT, High16Bits(offset));
1606 Jic(AT, Low16Bits(offset));
1607 break;
1608 case Branch::kR6LongCondBranch:
1609 EmitBcondc(Branch::OppositeCondition(condition), lhs, rhs, 2);
1610 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1611 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1612 Auipc(AT, High16Bits(offset));
1613 Jic(AT, Low16Bits(offset));
1614 break;
1615 case Branch::kR6LongCall:
1616 offset += (offset & 0x8000) << 1; // Account for sign extension in addiu.
1617 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1618 Auipc(lhs, High16Bits(offset));
1619 Addiu(lhs, lhs, Low16Bits(offset));
1620 Jialc(lhs, 0);
1621 break;
1622 }
1623 CHECK_EQ(overwrite_location_, branch->GetEndLocation());
1624 CHECK_LT(branch->GetSize(), static_cast<uint32_t>(Branch::kMaxBranchSize));
1625}
1626
1627void MipsAssembler::B(MipsLabel* label) {
1628 Buncond(label);
1629}
1630
1631void MipsAssembler::Jalr(MipsLabel* label, Register indirect_reg) {
1632 Call(label, indirect_reg);
1633}
1634
1635void MipsAssembler::Beq(Register rs, Register rt, MipsLabel* label) {
1636 Bcond(label, kCondEQ, rs, rt);
1637}
1638
1639void MipsAssembler::Bne(Register rs, Register rt, MipsLabel* label) {
1640 Bcond(label, kCondNE, rs, rt);
1641}
1642
1643void MipsAssembler::Beqz(Register rt, MipsLabel* label) {
1644 Bcond(label, kCondEQZ, rt);
1645}
1646
1647void MipsAssembler::Bnez(Register rt, MipsLabel* label) {
1648 Bcond(label, kCondNEZ, rt);
1649}
1650
1651void MipsAssembler::Bltz(Register rt, MipsLabel* label) {
1652 Bcond(label, kCondLTZ, rt);
1653}
1654
1655void MipsAssembler::Bgez(Register rt, MipsLabel* label) {
1656 Bcond(label, kCondGEZ, rt);
1657}
1658
1659void MipsAssembler::Blez(Register rt, MipsLabel* label) {
1660 Bcond(label, kCondLEZ, rt);
1661}
1662
1663void MipsAssembler::Bgtz(Register rt, MipsLabel* label) {
1664 Bcond(label, kCondGTZ, rt);
1665}
1666
1667void MipsAssembler::Blt(Register rs, Register rt, MipsLabel* label) {
1668 if (IsR6()) {
1669 Bcond(label, kCondLT, rs, rt);
1670 } else if (!Branch::IsNop(kCondLT, rs, rt)) {
1671 // Synthesize the instruction (not available on R2).
1672 Slt(AT, rs, rt);
1673 Bnez(AT, label);
1674 }
1675}
1676
1677void MipsAssembler::Bge(Register rs, Register rt, MipsLabel* label) {
1678 if (IsR6()) {
1679 Bcond(label, kCondGE, rs, rt);
1680 } else if (Branch::IsUncond(kCondGE, rs, rt)) {
1681 B(label);
1682 } else {
1683 // Synthesize the instruction (not available on R2).
1684 Slt(AT, rs, rt);
1685 Beqz(AT, label);
1686 }
1687}
1688
1689void MipsAssembler::Bltu(Register rs, Register rt, MipsLabel* label) {
1690 if (IsR6()) {
1691 Bcond(label, kCondLTU, rs, rt);
1692 } else if (!Branch::IsNop(kCondLTU, rs, rt)) {
1693 // Synthesize the instruction (not available on R2).
1694 Sltu(AT, rs, rt);
1695 Bnez(AT, label);
1696 }
1697}
1698
1699void MipsAssembler::Bgeu(Register rs, Register rt, MipsLabel* label) {
1700 if (IsR6()) {
1701 Bcond(label, kCondGEU, rs, rt);
1702 } else if (Branch::IsUncond(kCondGEU, rs, rt)) {
1703 B(label);
1704 } else {
1705 // Synthesize the instruction (not available on R2).
1706 Sltu(AT, rs, rt);
1707 Beqz(AT, label);
jeffhao7fbee072012-08-24 17:56:54 -07001708 }
1709}
1710
1711void MipsAssembler::LoadFromOffset(LoadOperandType type, Register reg, Register base,
1712 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001713 // IsInt<16> must be passed a signed value.
1714 if (!IsInt<16>(offset) ||
1715 (type == kLoadDoubleword && !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
jeffhao7fbee072012-08-24 17:56:54 -07001722 switch (type) {
1723 case kLoadSignedByte:
1724 Lb(reg, base, offset);
1725 break;
1726 case kLoadUnsignedByte:
1727 Lbu(reg, base, offset);
1728 break;
1729 case kLoadSignedHalfword:
1730 Lh(reg, base, offset);
1731 break;
1732 case kLoadUnsignedHalfword:
1733 Lhu(reg, base, offset);
1734 break;
1735 case kLoadWord:
1736 Lw(reg, base, offset);
1737 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001738 case kLoadDoubleword:
1739 if (reg == base) {
1740 // This will clobber the base when loading the lower register. Since we have to load the
1741 // higher register as well, this will fail. Solution: reverse the order.
1742 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
1743 Lw(reg, base, offset);
1744 } else {
1745 Lw(reg, base, offset);
1746 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
1747 }
jeffhao7fbee072012-08-24 17:56:54 -07001748 break;
1749 default:
1750 LOG(FATAL) << "UNREACHABLE";
1751 }
1752}
1753
1754void MipsAssembler::LoadSFromOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001755 if (!IsInt<16>(offset)) {
1756 LoadConst32(AT, offset);
1757 Addu(AT, AT, base);
1758 base = AT;
1759 offset = 0;
1760 }
1761
jeffhao7fbee072012-08-24 17:56:54 -07001762 Lwc1(reg, base, offset);
1763}
1764
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001765void MipsAssembler::LoadDFromOffset(FRegister reg, Register base, int32_t offset) {
1766 // IsInt<16> must be passed a signed value.
1767 if (!IsInt<16>(offset) ||
1768 (!IsAligned<kMipsDoublewordSize>(offset) &&
1769 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1770 LoadConst32(AT, offset);
1771 Addu(AT, AT, base);
1772 base = AT;
1773 offset = 0;
1774 }
1775
1776 if (offset & 0x7) {
1777 if (Is32BitFPU()) {
1778 Lwc1(reg, base, offset);
1779 Lwc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
1780 } else {
1781 // 64-bit FPU.
1782 Lwc1(reg, base, offset);
1783 Lw(T8, base, offset + kMipsWordSize);
1784 Mthc1(T8, reg);
1785 }
1786 } else {
1787 Ldc1(reg, base, offset);
1788 }
1789}
1790
1791void MipsAssembler::EmitLoad(ManagedRegister m_dst, Register src_register, int32_t src_offset,
1792 size_t size) {
1793 MipsManagedRegister dst = m_dst.AsMips();
1794 if (dst.IsNoRegister()) {
1795 CHECK_EQ(0u, size) << dst;
1796 } else if (dst.IsCoreRegister()) {
1797 CHECK_EQ(kMipsWordSize, size) << dst;
1798 LoadFromOffset(kLoadWord, dst.AsCoreRegister(), src_register, src_offset);
1799 } else if (dst.IsRegisterPair()) {
1800 CHECK_EQ(kMipsDoublewordSize, size) << dst;
1801 LoadFromOffset(kLoadDoubleword, dst.AsRegisterPairLow(), src_register, src_offset);
1802 } else if (dst.IsFRegister()) {
1803 if (size == kMipsWordSize) {
1804 LoadSFromOffset(dst.AsFRegister(), src_register, src_offset);
1805 } else {
1806 CHECK_EQ(kMipsDoublewordSize, size) << dst;
1807 LoadDFromOffset(dst.AsFRegister(), src_register, src_offset);
1808 }
1809 }
jeffhao7fbee072012-08-24 17:56:54 -07001810}
1811
1812void MipsAssembler::StoreToOffset(StoreOperandType type, Register reg, Register base,
1813 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001814 // IsInt<16> must be passed a signed value.
1815 if (!IsInt<16>(offset) ||
1816 (type == kStoreDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1817 LoadConst32(AT, offset);
1818 Addu(AT, AT, base);
1819 base = AT;
1820 offset = 0;
1821 }
1822
jeffhao7fbee072012-08-24 17:56:54 -07001823 switch (type) {
1824 case kStoreByte:
1825 Sb(reg, base, offset);
1826 break;
1827 case kStoreHalfword:
1828 Sh(reg, base, offset);
1829 break;
1830 case kStoreWord:
1831 Sw(reg, base, offset);
1832 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001833 case kStoreDoubleword:
1834 CHECK_NE(reg, base);
1835 CHECK_NE(static_cast<Register>(reg + 1), base);
1836 Sw(reg, base, offset);
1837 Sw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001838 break;
1839 default:
1840 LOG(FATAL) << "UNREACHABLE";
1841 }
1842}
1843
Goran Jakovljevicff734982015-08-24 12:58:55 +00001844void MipsAssembler::StoreSToOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001845 if (!IsInt<16>(offset)) {
1846 LoadConst32(AT, offset);
1847 Addu(AT, AT, base);
1848 base = AT;
1849 offset = 0;
1850 }
1851
jeffhao7fbee072012-08-24 17:56:54 -07001852 Swc1(reg, base, offset);
1853}
1854
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001855void MipsAssembler::StoreDToOffset(FRegister reg, Register base, int32_t offset) {
1856 // IsInt<16> must be passed a signed value.
1857 if (!IsInt<16>(offset) ||
1858 (!IsAligned<kMipsDoublewordSize>(offset) &&
1859 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1860 LoadConst32(AT, offset);
1861 Addu(AT, AT, base);
1862 base = AT;
1863 offset = 0;
1864 }
1865
1866 if (offset & 0x7) {
1867 if (Is32BitFPU()) {
1868 Swc1(reg, base, offset);
1869 Swc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
1870 } else {
1871 // 64-bit FPU.
1872 Mfhc1(T8, reg);
1873 Swc1(reg, base, offset);
1874 Sw(T8, base, offset + kMipsWordSize);
1875 }
1876 } else {
1877 Sdc1(reg, base, offset);
1878 }
jeffhao7fbee072012-08-24 17:56:54 -07001879}
1880
David Srbeckydd973932015-04-07 20:29:48 +01001881static dwarf::Reg DWARFReg(Register reg) {
1882 return dwarf::Reg::MipsCore(static_cast<int>(reg));
1883}
1884
Ian Rogers790a6b72014-04-01 10:36:00 -07001885constexpr size_t kFramePointerSize = 4;
1886
jeffhao7fbee072012-08-24 17:56:54 -07001887void MipsAssembler::BuildFrame(size_t frame_size, ManagedRegister method_reg,
1888 const std::vector<ManagedRegister>& callee_save_regs,
Dmitry Petrochenkofca82202014-03-21 11:21:37 +07001889 const ManagedRegisterEntrySpills& entry_spills) {
jeffhao7fbee072012-08-24 17:56:54 -07001890 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001891 DCHECK(!overwriting_);
jeffhao7fbee072012-08-24 17:56:54 -07001892
1893 // Increase frame to required size.
1894 IncreaseFrameSize(frame_size);
1895
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001896 // Push callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07001897 int stack_offset = frame_size - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001898 StoreToOffset(kStoreWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001899 cfi_.RelOffset(DWARFReg(RA), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07001900 for (int i = callee_save_regs.size() - 1; i >= 0; --i) {
Ian Rogers790a6b72014-04-01 10:36:00 -07001901 stack_offset -= kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001902 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
1903 StoreToOffset(kStoreWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001904 cfi_.RelOffset(DWARFReg(reg), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07001905 }
1906
1907 // Write out Method*.
1908 StoreToOffset(kStoreWord, method_reg.AsMips().AsCoreRegister(), SP, 0);
1909
1910 // Write out entry spills.
Goran Jakovljevicff734982015-08-24 12:58:55 +00001911 int32_t offset = frame_size + kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001912 for (size_t i = 0; i < entry_spills.size(); ++i) {
Goran Jakovljevicff734982015-08-24 12:58:55 +00001913 MipsManagedRegister reg = entry_spills.at(i).AsMips();
1914 if (reg.IsNoRegister()) {
1915 ManagedRegisterSpill spill = entry_spills.at(i);
1916 offset += spill.getSize();
1917 } else if (reg.IsCoreRegister()) {
1918 StoreToOffset(kStoreWord, reg.AsCoreRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001919 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001920 } else if (reg.IsFRegister()) {
1921 StoreSToOffset(reg.AsFRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001922 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001923 } else if (reg.IsDRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001924 StoreDToOffset(reg.AsOverlappingDRegisterLow(), SP, offset);
1925 offset += kMipsDoublewordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001926 }
jeffhao7fbee072012-08-24 17:56:54 -07001927 }
1928}
1929
1930void MipsAssembler::RemoveFrame(size_t frame_size,
1931 const std::vector<ManagedRegister>& callee_save_regs) {
1932 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001933 DCHECK(!overwriting_);
David Srbeckydd973932015-04-07 20:29:48 +01001934 cfi_.RememberState();
jeffhao7fbee072012-08-24 17:56:54 -07001935
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001936 // Pop callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07001937 int stack_offset = frame_size - (callee_save_regs.size() * kFramePointerSize) - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001938 for (size_t i = 0; i < callee_save_regs.size(); ++i) {
1939 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
1940 LoadFromOffset(kLoadWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001941 cfi_.Restore(DWARFReg(reg));
Ian Rogers790a6b72014-04-01 10:36:00 -07001942 stack_offset += kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001943 }
1944 LoadFromOffset(kLoadWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001945 cfi_.Restore(DWARFReg(RA));
jeffhao7fbee072012-08-24 17:56:54 -07001946
1947 // Decrease frame to required size.
1948 DecreaseFrameSize(frame_size);
jeffhao07030602012-09-26 14:33:14 -07001949
1950 // Then jump to the return address.
1951 Jr(RA);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001952 Nop();
David Srbeckydd973932015-04-07 20:29:48 +01001953
1954 // The CFI should be restored for any code that follows the exit block.
1955 cfi_.RestoreState();
1956 cfi_.DefCFAOffset(frame_size);
jeffhao7fbee072012-08-24 17:56:54 -07001957}
1958
1959void MipsAssembler::IncreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001960 CHECK_ALIGNED(adjust, kFramePointerSize);
1961 Addiu32(SP, SP, -adjust);
David Srbeckydd973932015-04-07 20:29:48 +01001962 cfi_.AdjustCFAOffset(adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001963 if (overwriting_) {
1964 cfi_.OverrideDelayedPC(overwrite_location_);
1965 }
jeffhao7fbee072012-08-24 17:56:54 -07001966}
1967
1968void MipsAssembler::DecreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001969 CHECK_ALIGNED(adjust, kFramePointerSize);
1970 Addiu32(SP, SP, adjust);
David Srbeckydd973932015-04-07 20:29:48 +01001971 cfi_.AdjustCFAOffset(-adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001972 if (overwriting_) {
1973 cfi_.OverrideDelayedPC(overwrite_location_);
1974 }
jeffhao7fbee072012-08-24 17:56:54 -07001975}
1976
1977void MipsAssembler::Store(FrameOffset dest, ManagedRegister msrc, size_t size) {
1978 MipsManagedRegister src = msrc.AsMips();
1979 if (src.IsNoRegister()) {
1980 CHECK_EQ(0u, size);
1981 } else if (src.IsCoreRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001982 CHECK_EQ(kMipsWordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07001983 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
1984 } else if (src.IsRegisterPair()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001985 CHECK_EQ(kMipsDoublewordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07001986 StoreToOffset(kStoreWord, src.AsRegisterPairLow(), SP, dest.Int32Value());
1987 StoreToOffset(kStoreWord, src.AsRegisterPairHigh(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001988 SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001989 } else if (src.IsFRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001990 if (size == kMipsWordSize) {
1991 StoreSToOffset(src.AsFRegister(), SP, dest.Int32Value());
1992 } else {
1993 CHECK_EQ(kMipsDoublewordSize, size);
1994 StoreDToOffset(src.AsFRegister(), SP, dest.Int32Value());
1995 }
jeffhao7fbee072012-08-24 17:56:54 -07001996 }
1997}
1998
1999void MipsAssembler::StoreRef(FrameOffset dest, ManagedRegister msrc) {
2000 MipsManagedRegister src = msrc.AsMips();
2001 CHECK(src.IsCoreRegister());
2002 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2003}
2004
2005void MipsAssembler::StoreRawPtr(FrameOffset dest, ManagedRegister msrc) {
2006 MipsManagedRegister src = msrc.AsMips();
2007 CHECK(src.IsCoreRegister());
2008 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2009}
2010
2011void MipsAssembler::StoreImmediateToFrame(FrameOffset dest, uint32_t imm,
2012 ManagedRegister mscratch) {
2013 MipsManagedRegister scratch = mscratch.AsMips();
2014 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002015 LoadConst32(scratch.AsCoreRegister(), imm);
jeffhao7fbee072012-08-24 17:56:54 -07002016 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2017}
2018
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002019void MipsAssembler::StoreImmediateToThread32(ThreadOffset<kMipsWordSize> dest, uint32_t imm,
jeffhao7fbee072012-08-24 17:56:54 -07002020 ManagedRegister mscratch) {
2021 MipsManagedRegister scratch = mscratch.AsMips();
2022 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002023 // Is this function even referenced anywhere else in the code?
2024 LoadConst32(scratch.AsCoreRegister(), imm);
2025 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), S1, dest.Int32Value());
2026}
2027
2028void MipsAssembler::StoreStackOffsetToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2029 FrameOffset fr_offs,
2030 ManagedRegister mscratch) {
2031 MipsManagedRegister scratch = mscratch.AsMips();
2032 CHECK(scratch.IsCoreRegister()) << scratch;
2033 Addiu32(scratch.AsCoreRegister(), SP, fr_offs.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002034 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2035 S1, thr_offs.Int32Value());
2036}
2037
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002038void MipsAssembler::StoreStackPointerToThread32(ThreadOffset<kMipsWordSize> thr_offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002039 StoreToOffset(kStoreWord, SP, S1, thr_offs.Int32Value());
2040}
2041
2042void MipsAssembler::StoreSpanning(FrameOffset dest, ManagedRegister msrc,
2043 FrameOffset in_off, ManagedRegister mscratch) {
2044 MipsManagedRegister src = msrc.AsMips();
2045 MipsManagedRegister scratch = mscratch.AsMips();
2046 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2047 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, in_off.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002048 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002049}
2050
2051void MipsAssembler::Load(ManagedRegister mdest, FrameOffset src, size_t size) {
2052 return EmitLoad(mdest, SP, src.Int32Value(), size);
2053}
2054
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002055void MipsAssembler::LoadFromThread32(ManagedRegister mdest,
2056 ThreadOffset<kMipsWordSize> src, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002057 return EmitLoad(mdest, S1, src.Int32Value(), size);
2058}
2059
2060void MipsAssembler::LoadRef(ManagedRegister mdest, FrameOffset src) {
2061 MipsManagedRegister dest = mdest.AsMips();
2062 CHECK(dest.IsCoreRegister());
2063 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), SP, src.Int32Value());
2064}
2065
Mathieu Chartiere401d142015-04-22 13:56:20 -07002066void MipsAssembler::LoadRef(ManagedRegister mdest, ManagedRegister base, MemberOffset offs,
Roland Levillain4d027112015-07-01 15:41:14 +01002067 bool unpoison_reference) {
jeffhao7fbee072012-08-24 17:56:54 -07002068 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002069 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002070 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2071 base.AsMips().AsCoreRegister(), offs.Int32Value());
Roland Levillain4d027112015-07-01 15:41:14 +01002072 if (kPoisonHeapReferences && unpoison_reference) {
Hiroshi Yamauchie63a7452014-02-27 14:44:36 -08002073 Subu(dest.AsCoreRegister(), ZERO, dest.AsCoreRegister());
2074 }
jeffhao7fbee072012-08-24 17:56:54 -07002075}
2076
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002077void MipsAssembler::LoadRawPtr(ManagedRegister mdest, ManagedRegister base, Offset offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002078 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002079 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002080 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2081 base.AsMips().AsCoreRegister(), offs.Int32Value());
2082}
2083
Ian Rogersdd7624d2014-03-14 17:43:00 -07002084void MipsAssembler::LoadRawPtrFromThread32(ManagedRegister mdest,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002085 ThreadOffset<kMipsWordSize> offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002086 MipsManagedRegister dest = mdest.AsMips();
2087 CHECK(dest.IsCoreRegister());
2088 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), S1, offs.Int32Value());
2089}
2090
2091void MipsAssembler::SignExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2092 UNIMPLEMENTED(FATAL) << "no sign extension necessary for mips";
2093}
2094
2095void MipsAssembler::ZeroExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2096 UNIMPLEMENTED(FATAL) << "no zero extension necessary for mips";
2097}
2098
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002099void MipsAssembler::Move(ManagedRegister mdest, ManagedRegister msrc, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002100 MipsManagedRegister dest = mdest.AsMips();
2101 MipsManagedRegister src = msrc.AsMips();
2102 if (!dest.Equals(src)) {
2103 if (dest.IsCoreRegister()) {
2104 CHECK(src.IsCoreRegister()) << src;
2105 Move(dest.AsCoreRegister(), src.AsCoreRegister());
2106 } else if (dest.IsFRegister()) {
2107 CHECK(src.IsFRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002108 if (size == kMipsWordSize) {
2109 MovS(dest.AsFRegister(), src.AsFRegister());
2110 } else {
2111 CHECK_EQ(kMipsDoublewordSize, size);
2112 MovD(dest.AsFRegister(), src.AsFRegister());
2113 }
jeffhao7fbee072012-08-24 17:56:54 -07002114 } else if (dest.IsDRegister()) {
2115 CHECK(src.IsDRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002116 MovD(dest.AsOverlappingDRegisterLow(), src.AsOverlappingDRegisterLow());
jeffhao7fbee072012-08-24 17:56:54 -07002117 } else {
2118 CHECK(dest.IsRegisterPair()) << dest;
2119 CHECK(src.IsRegisterPair()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002120 // Ensure that the first move doesn't clobber the input of the second.
jeffhao7fbee072012-08-24 17:56:54 -07002121 if (src.AsRegisterPairHigh() != dest.AsRegisterPairLow()) {
2122 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2123 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2124 } else {
2125 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2126 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2127 }
2128 }
2129 }
2130}
2131
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002132void MipsAssembler::CopyRef(FrameOffset dest, FrameOffset src, ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002133 MipsManagedRegister scratch = mscratch.AsMips();
2134 CHECK(scratch.IsCoreRegister()) << scratch;
2135 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2136 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2137}
2138
Ian Rogersdd7624d2014-03-14 17:43:00 -07002139void MipsAssembler::CopyRawPtrFromThread32(FrameOffset fr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002140 ThreadOffset<kMipsWordSize> thr_offs,
2141 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002142 MipsManagedRegister scratch = mscratch.AsMips();
2143 CHECK(scratch.IsCoreRegister()) << scratch;
2144 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2145 S1, thr_offs.Int32Value());
2146 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2147 SP, fr_offs.Int32Value());
2148}
2149
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002150void MipsAssembler::CopyRawPtrToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2151 FrameOffset fr_offs,
2152 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002153 MipsManagedRegister scratch = mscratch.AsMips();
2154 CHECK(scratch.IsCoreRegister()) << scratch;
2155 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2156 SP, fr_offs.Int32Value());
2157 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2158 S1, thr_offs.Int32Value());
2159}
2160
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002161void MipsAssembler::Copy(FrameOffset dest, FrameOffset src, ManagedRegister mscratch, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002162 MipsManagedRegister scratch = mscratch.AsMips();
2163 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002164 CHECK(size == kMipsWordSize || size == kMipsDoublewordSize) << size;
2165 if (size == kMipsWordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002166 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2167 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002168 } else if (size == kMipsDoublewordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002169 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2170 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002171 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value() + kMipsWordSize);
2172 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002173 }
2174}
2175
2176void MipsAssembler::Copy(FrameOffset dest, ManagedRegister src_base, Offset src_offset,
2177 ManagedRegister mscratch, size_t size) {
2178 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002179 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002180 LoadFromOffset(kLoadWord, scratch, src_base.AsMips().AsCoreRegister(), src_offset.Int32Value());
2181 StoreToOffset(kStoreWord, scratch, SP, dest.Int32Value());
2182}
2183
2184void MipsAssembler::Copy(ManagedRegister dest_base, Offset dest_offset, FrameOffset src,
2185 ManagedRegister mscratch, size_t size) {
2186 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002187 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002188 LoadFromOffset(kLoadWord, scratch, SP, src.Int32Value());
2189 StoreToOffset(kStoreWord, scratch, dest_base.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2190}
2191
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002192void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2193 FrameOffset src_base ATTRIBUTE_UNUSED,
2194 Offset src_offset ATTRIBUTE_UNUSED,
2195 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2196 size_t size ATTRIBUTE_UNUSED) {
2197 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002198}
2199
2200void MipsAssembler::Copy(ManagedRegister dest, Offset dest_offset,
2201 ManagedRegister src, Offset src_offset,
2202 ManagedRegister mscratch, size_t size) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002203 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002204 Register scratch = mscratch.AsMips().AsCoreRegister();
2205 LoadFromOffset(kLoadWord, scratch, src.AsMips().AsCoreRegister(), src_offset.Int32Value());
2206 StoreToOffset(kStoreWord, scratch, dest.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2207}
2208
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002209void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2210 Offset dest_offset ATTRIBUTE_UNUSED,
2211 FrameOffset src ATTRIBUTE_UNUSED,
2212 Offset src_offset ATTRIBUTE_UNUSED,
2213 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2214 size_t size ATTRIBUTE_UNUSED) {
2215 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002216}
2217
2218void MipsAssembler::MemoryBarrier(ManagedRegister) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002219 // TODO: sync?
2220 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002221}
2222
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002223void MipsAssembler::CreateHandleScopeEntry(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002224 FrameOffset handle_scope_offset,
2225 ManagedRegister min_reg,
2226 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002227 MipsManagedRegister out_reg = mout_reg.AsMips();
2228 MipsManagedRegister in_reg = min_reg.AsMips();
2229 CHECK(in_reg.IsNoRegister() || in_reg.IsCoreRegister()) << in_reg;
2230 CHECK(out_reg.IsCoreRegister()) << out_reg;
2231 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002232 MipsLabel null_arg;
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002233 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2234 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002235 // E.g. out_reg = (handle == 0) ? 0 : (SP+handle_offset).
jeffhao7fbee072012-08-24 17:56:54 -07002236 if (in_reg.IsNoRegister()) {
2237 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002238 SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002239 in_reg = out_reg;
2240 }
2241 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002242 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002243 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002244 Beqz(in_reg.AsCoreRegister(), &null_arg);
2245 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2246 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002247 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002248 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002249 }
2250}
2251
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002252void MipsAssembler::CreateHandleScopeEntry(FrameOffset out_off,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002253 FrameOffset handle_scope_offset,
2254 ManagedRegister mscratch,
2255 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002256 MipsManagedRegister scratch = mscratch.AsMips();
2257 CHECK(scratch.IsCoreRegister()) << scratch;
2258 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002259 MipsLabel null_arg;
2260 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002261 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2262 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002263 // E.g. scratch = (scratch == 0) ? 0 : (SP+handle_scope_offset).
2264 Beqz(scratch.AsCoreRegister(), &null_arg);
2265 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2266 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002267 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002268 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002269 }
2270 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, out_off.Int32Value());
2271}
2272
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002273// Given a handle scope entry, load the associated reference.
2274void MipsAssembler::LoadReferenceFromHandleScope(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002275 ManagedRegister min_reg) {
jeffhao7fbee072012-08-24 17:56:54 -07002276 MipsManagedRegister out_reg = mout_reg.AsMips();
2277 MipsManagedRegister in_reg = min_reg.AsMips();
2278 CHECK(out_reg.IsCoreRegister()) << out_reg;
2279 CHECK(in_reg.IsCoreRegister()) << in_reg;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002280 MipsLabel null_arg;
jeffhao7fbee072012-08-24 17:56:54 -07002281 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002282 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002283 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002284 Beqz(in_reg.AsCoreRegister(), &null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002285 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
2286 in_reg.AsCoreRegister(), 0);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002287 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002288}
2289
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002290void MipsAssembler::VerifyObject(ManagedRegister src ATTRIBUTE_UNUSED,
2291 bool could_be_null ATTRIBUTE_UNUSED) {
2292 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002293}
2294
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002295void MipsAssembler::VerifyObject(FrameOffset src ATTRIBUTE_UNUSED,
2296 bool could_be_null ATTRIBUTE_UNUSED) {
2297 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002298}
2299
2300void MipsAssembler::Call(ManagedRegister mbase, Offset offset, ManagedRegister mscratch) {
2301 MipsManagedRegister base = mbase.AsMips();
2302 MipsManagedRegister scratch = mscratch.AsMips();
2303 CHECK(base.IsCoreRegister()) << base;
2304 CHECK(scratch.IsCoreRegister()) << scratch;
2305 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2306 base.AsCoreRegister(), offset.Int32Value());
2307 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002308 Nop();
2309 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002310}
2311
2312void MipsAssembler::Call(FrameOffset base, Offset offset, ManagedRegister mscratch) {
2313 MipsManagedRegister scratch = mscratch.AsMips();
2314 CHECK(scratch.IsCoreRegister()) << scratch;
2315 // Call *(*(SP + base) + offset)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002316 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, base.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002317 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2318 scratch.AsCoreRegister(), offset.Int32Value());
2319 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002320 Nop();
2321 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002322}
2323
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002324void MipsAssembler::CallFromThread32(ThreadOffset<kMipsWordSize> offset ATTRIBUTE_UNUSED,
2325 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
Ian Rogers468532e2013-08-05 10:56:33 -07002326 UNIMPLEMENTED(FATAL) << "no mips implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002327}
2328
2329void MipsAssembler::GetCurrentThread(ManagedRegister tr) {
2330 Move(tr.AsMips().AsCoreRegister(), S1);
2331}
2332
2333void MipsAssembler::GetCurrentThread(FrameOffset offset,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002334 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
jeffhao7fbee072012-08-24 17:56:54 -07002335 StoreToOffset(kStoreWord, S1, SP, offset.Int32Value());
2336}
2337
jeffhao7fbee072012-08-24 17:56:54 -07002338void MipsAssembler::ExceptionPoll(ManagedRegister mscratch, size_t stack_adjust) {
2339 MipsManagedRegister scratch = mscratch.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002340 exception_blocks_.emplace_back(scratch, stack_adjust);
jeffhao7fbee072012-08-24 17:56:54 -07002341 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002342 S1, Thread::ExceptionOffset<kMipsWordSize>().Int32Value());
2343 // TODO: on MIPS32R6 prefer Bnezc(scratch.AsCoreRegister(), slow.Entry());
2344 // as the NAL instruction (occurring in long R2 branches) may become deprecated.
2345 // For now use common for R2 and R6 instructions as this code must execute on both.
2346 Bnez(scratch.AsCoreRegister(), exception_blocks_.back().Entry());
jeffhao7fbee072012-08-24 17:56:54 -07002347}
2348
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002349void MipsAssembler::EmitExceptionPoll(MipsExceptionSlowPath* exception) {
2350 Bind(exception->Entry());
2351 if (exception->stack_adjust_ != 0) { // Fix up the frame.
2352 DecreaseFrameSize(exception->stack_adjust_);
jeffhao7fbee072012-08-24 17:56:54 -07002353 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002354 // Pass exception object as argument.
2355 // Don't care about preserving A0 as this call won't return.
2356 CheckEntrypointTypes<kQuickDeliverException, void, mirror::Object*>();
2357 Move(A0, exception->scratch_.AsCoreRegister());
2358 // Set up call to Thread::Current()->pDeliverException.
2359 LoadFromOffset(kLoadWord, T9, S1,
2360 QUICK_ENTRYPOINT_OFFSET(kMipsWordSize, pDeliverException).Int32Value());
2361 Jr(T9);
2362 Nop();
2363
2364 // Call never returns.
2365 Break();
jeffhao7fbee072012-08-24 17:56:54 -07002366}
2367
2368} // namespace mips
2369} // namespace art