blob: 42f21e603d2e2f30136e62d79c27c292487d3eb2 [file] [log] [blame]
jeffhao7fbee072012-08-24 17:56:54 -07001/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "assembler_mips.h"
18
Vladimir Marko80afd022015-05-19 18:08:00 +010019#include "base/bit_utils.h"
Elliott Hughes1aa246d2012-12-13 09:29:36 -080020#include "base/casts.h"
Ian Rogers166db042013-07-26 12:05:57 -070021#include "entrypoints/quick/quick_entrypoints.h"
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020022#include "entrypoints/quick/quick_entrypoints_enum.h"
jeffhao7fbee072012-08-24 17:56:54 -070023#include "memory_region.h"
jeffhao7fbee072012-08-24 17:56:54 -070024#include "thread.h"
25
26namespace art {
27namespace mips {
jeffhao7fbee072012-08-24 17:56:54 -070028
jeffhao7fbee072012-08-24 17:56:54 -070029std::ostream& operator<<(std::ostream& os, const DRegister& rhs) {
30 if (rhs >= D0 && rhs < kNumberOfDRegisters) {
31 os << "d" << static_cast<int>(rhs);
32 } else {
33 os << "DRegister[" << static_cast<int>(rhs) << "]";
34 }
35 return os;
36}
37
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020038void MipsAssembler::FinalizeCode() {
39 for (auto& exception_block : exception_blocks_) {
40 EmitExceptionPoll(&exception_block);
41 }
42 PromoteBranches();
43}
44
45void MipsAssembler::FinalizeInstructions(const MemoryRegion& region) {
Vladimir Marko10ef6942015-10-22 15:25:54 +010046 size_t number_of_delayed_adjust_pcs = cfi().NumberOfDelayedAdvancePCs();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020047 EmitBranches();
48 Assembler::FinalizeInstructions(region);
Vladimir Marko10ef6942015-10-22 15:25:54 +010049 PatchCFI(number_of_delayed_adjust_pcs);
50}
51
52void MipsAssembler::PatchCFI(size_t number_of_delayed_adjust_pcs) {
53 if (cfi().NumberOfDelayedAdvancePCs() == 0u) {
54 DCHECK_EQ(number_of_delayed_adjust_pcs, 0u);
55 return;
56 }
57
58 typedef DebugFrameOpCodeWriterForAssembler::DelayedAdvancePC DelayedAdvancePC;
59 const auto data = cfi().ReleaseStreamAndPrepareForDelayedAdvancePC();
60 const std::vector<uint8_t>& old_stream = data.first;
61 const std::vector<DelayedAdvancePC>& advances = data.second;
62
63 // PCs recorded before EmitBranches() need to be adjusted.
64 // PCs recorded during EmitBranches() are already adjusted.
65 // Both ranges are separately sorted but they may overlap.
66 if (kIsDebugBuild) {
67 auto cmp = [](const DelayedAdvancePC& lhs, const DelayedAdvancePC& rhs) {
68 return lhs.pc < rhs.pc;
69 };
70 CHECK(std::is_sorted(advances.begin(), advances.begin() + number_of_delayed_adjust_pcs, cmp));
71 CHECK(std::is_sorted(advances.begin() + number_of_delayed_adjust_pcs, advances.end(), cmp));
72 }
73
74 // Append initial CFI data if any.
75 size_t size = advances.size();
76 DCHECK_NE(size, 0u);
77 cfi().AppendRawData(old_stream, 0u, advances[0].stream_pos);
78 // Emit PC adjustments interleaved with the old CFI stream.
79 size_t adjust_pos = 0u;
80 size_t late_emit_pos = number_of_delayed_adjust_pcs;
81 while (adjust_pos != number_of_delayed_adjust_pcs || late_emit_pos != size) {
82 size_t adjusted_pc = (adjust_pos != number_of_delayed_adjust_pcs)
83 ? GetAdjustedPosition(advances[adjust_pos].pc)
84 : static_cast<size_t>(-1);
85 size_t late_emit_pc = (late_emit_pos != size)
86 ? advances[late_emit_pos].pc
87 : static_cast<size_t>(-1);
88 size_t advance_pc = std::min(adjusted_pc, late_emit_pc);
89 DCHECK_NE(advance_pc, static_cast<size_t>(-1));
90 size_t entry = (adjusted_pc <= late_emit_pc) ? adjust_pos : late_emit_pos;
91 if (adjusted_pc <= late_emit_pc) {
92 ++adjust_pos;
93 } else {
94 ++late_emit_pos;
95 }
96 cfi().AdvancePC(advance_pc);
97 size_t end_pos = (entry + 1u == size) ? old_stream.size() : advances[entry + 1u].stream_pos;
98 cfi().AppendRawData(old_stream, advances[entry].stream_pos, end_pos);
99 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200100}
101
102void MipsAssembler::EmitBranches() {
103 CHECK(!overwriting_);
104 // Switch from appending instructions at the end of the buffer to overwriting
105 // existing instructions (branch placeholders) in the buffer.
106 overwriting_ = true;
107 for (auto& branch : branches_) {
108 EmitBranch(&branch);
109 }
110 overwriting_ = false;
111}
112
113void MipsAssembler::Emit(uint32_t value) {
114 if (overwriting_) {
115 // Branches to labels are emitted into their placeholders here.
116 buffer_.Store<uint32_t>(overwrite_location_, value);
117 overwrite_location_ += sizeof(uint32_t);
118 } else {
119 // Other instructions are simply appended at the end here.
120 AssemblerBuffer::EnsureCapacity ensured(&buffer_);
121 buffer_.Emit<uint32_t>(value);
122 }
jeffhao7fbee072012-08-24 17:56:54 -0700123}
124
125void MipsAssembler::EmitR(int opcode, Register rs, Register rt, Register rd, int shamt, int funct) {
126 CHECK_NE(rs, kNoRegister);
127 CHECK_NE(rt, kNoRegister);
128 CHECK_NE(rd, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200129 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
130 static_cast<uint32_t>(rs) << kRsShift |
131 static_cast<uint32_t>(rt) << kRtShift |
132 static_cast<uint32_t>(rd) << kRdShift |
133 shamt << kShamtShift |
134 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700135 Emit(encoding);
136}
137
138void MipsAssembler::EmitI(int opcode, Register rs, Register rt, uint16_t imm) {
139 CHECK_NE(rs, kNoRegister);
140 CHECK_NE(rt, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200141 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
142 static_cast<uint32_t>(rs) << kRsShift |
143 static_cast<uint32_t>(rt) << kRtShift |
144 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700145 Emit(encoding);
146}
147
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200148void MipsAssembler::EmitI21(int opcode, Register rs, uint32_t imm21) {
149 CHECK_NE(rs, kNoRegister);
150 CHECK(IsUint<21>(imm21)) << imm21;
151 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
152 static_cast<uint32_t>(rs) << kRsShift |
153 imm21;
jeffhao7fbee072012-08-24 17:56:54 -0700154 Emit(encoding);
155}
156
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200157void MipsAssembler::EmitI26(int opcode, uint32_t imm26) {
158 CHECK(IsUint<26>(imm26)) << imm26;
159 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift | imm26;
160 Emit(encoding);
161}
162
163void MipsAssembler::EmitFR(int opcode, int fmt, FRegister ft, FRegister fs, FRegister fd,
164 int funct) {
jeffhao7fbee072012-08-24 17:56:54 -0700165 CHECK_NE(ft, kNoFRegister);
166 CHECK_NE(fs, kNoFRegister);
167 CHECK_NE(fd, kNoFRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200168 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
169 fmt << kFmtShift |
170 static_cast<uint32_t>(ft) << kFtShift |
171 static_cast<uint32_t>(fs) << kFsShift |
172 static_cast<uint32_t>(fd) << kFdShift |
173 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700174 Emit(encoding);
175}
176
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200177void MipsAssembler::EmitFI(int opcode, int fmt, FRegister ft, uint16_t imm) {
178 CHECK_NE(ft, kNoFRegister);
179 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
180 fmt << kFmtShift |
181 static_cast<uint32_t>(ft) << kFtShift |
182 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700183 Emit(encoding);
184}
185
jeffhao7fbee072012-08-24 17:56:54 -0700186void MipsAssembler::Addu(Register rd, Register rs, Register rt) {
187 EmitR(0, rs, rt, rd, 0, 0x21);
188}
189
jeffhao7fbee072012-08-24 17:56:54 -0700190void MipsAssembler::Addiu(Register rt, Register rs, uint16_t imm16) {
191 EmitI(0x9, rs, rt, imm16);
192}
193
jeffhao7fbee072012-08-24 17:56:54 -0700194void MipsAssembler::Subu(Register rd, Register rs, Register rt) {
195 EmitR(0, rs, rt, rd, 0, 0x23);
196}
197
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200198void MipsAssembler::MultR2(Register rs, Register rt) {
199 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700200 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x18);
201}
202
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200203void MipsAssembler::MultuR2(Register rs, Register rt) {
204 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700205 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x19);
206}
207
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200208void MipsAssembler::DivR2(Register rs, Register rt) {
209 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700210 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1a);
211}
212
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200213void MipsAssembler::DivuR2(Register rs, Register rt) {
214 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700215 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1b);
216}
217
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200218void MipsAssembler::MulR2(Register rd, Register rs, Register rt) {
219 CHECK(!IsR6());
220 EmitR(0x1c, rs, rt, rd, 0, 2);
221}
222
223void MipsAssembler::DivR2(Register rd, Register rs, Register rt) {
224 CHECK(!IsR6());
225 DivR2(rs, rt);
226 Mflo(rd);
227}
228
229void MipsAssembler::ModR2(Register rd, Register rs, Register rt) {
230 CHECK(!IsR6());
231 DivR2(rs, rt);
232 Mfhi(rd);
233}
234
235void MipsAssembler::DivuR2(Register rd, Register rs, Register rt) {
236 CHECK(!IsR6());
237 DivuR2(rs, rt);
238 Mflo(rd);
239}
240
241void MipsAssembler::ModuR2(Register rd, Register rs, Register rt) {
242 CHECK(!IsR6());
243 DivuR2(rs, rt);
244 Mfhi(rd);
245}
246
247void MipsAssembler::MulR6(Register rd, Register rs, Register rt) {
248 CHECK(IsR6());
249 EmitR(0, rs, rt, rd, 2, 0x18);
250}
251
252void MipsAssembler::MuhuR6(Register rd, Register rs, Register rt) {
253 CHECK(IsR6());
254 EmitR(0, rs, rt, rd, 3, 0x19);
255}
256
257void MipsAssembler::DivR6(Register rd, Register rs, Register rt) {
258 CHECK(IsR6());
259 EmitR(0, rs, rt, rd, 2, 0x1a);
260}
261
262void MipsAssembler::ModR6(Register rd, Register rs, Register rt) {
263 CHECK(IsR6());
264 EmitR(0, rs, rt, rd, 3, 0x1a);
265}
266
267void MipsAssembler::DivuR6(Register rd, Register rs, Register rt) {
268 CHECK(IsR6());
269 EmitR(0, rs, rt, rd, 2, 0x1b);
270}
271
272void MipsAssembler::ModuR6(Register rd, Register rs, Register rt) {
273 CHECK(IsR6());
274 EmitR(0, rs, rt, rd, 3, 0x1b);
275}
276
jeffhao7fbee072012-08-24 17:56:54 -0700277void MipsAssembler::And(Register rd, Register rs, Register rt) {
278 EmitR(0, rs, rt, rd, 0, 0x24);
279}
280
281void MipsAssembler::Andi(Register rt, Register rs, uint16_t imm16) {
282 EmitI(0xc, rs, rt, imm16);
283}
284
285void MipsAssembler::Or(Register rd, Register rs, Register rt) {
286 EmitR(0, rs, rt, rd, 0, 0x25);
287}
288
289void MipsAssembler::Ori(Register rt, Register rs, uint16_t imm16) {
290 EmitI(0xd, rs, rt, imm16);
291}
292
293void MipsAssembler::Xor(Register rd, Register rs, Register rt) {
294 EmitR(0, rs, rt, rd, 0, 0x26);
295}
296
297void MipsAssembler::Xori(Register rt, Register rs, uint16_t imm16) {
298 EmitI(0xe, rs, rt, imm16);
299}
300
301void MipsAssembler::Nor(Register rd, Register rs, Register rt) {
302 EmitR(0, rs, rt, rd, 0, 0x27);
303}
304
Chris Larsene3845472015-11-18 12:27:15 -0800305void MipsAssembler::Movz(Register rd, Register rs, Register rt) {
306 CHECK(!IsR6());
307 EmitR(0, rs, rt, rd, 0, 0x0A);
308}
309
310void MipsAssembler::Movn(Register rd, Register rs, Register rt) {
311 CHECK(!IsR6());
312 EmitR(0, rs, rt, rd, 0, 0x0B);
313}
314
315void MipsAssembler::Seleqz(Register rd, Register rs, Register rt) {
316 CHECK(IsR6());
317 EmitR(0, rs, rt, rd, 0, 0x35);
318}
319
320void MipsAssembler::Selnez(Register rd, Register rs, Register rt) {
321 CHECK(IsR6());
322 EmitR(0, rs, rt, rd, 0, 0x37);
323}
324
325void MipsAssembler::ClzR6(Register rd, Register rs) {
326 CHECK(IsR6());
327 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x10);
328}
329
330void MipsAssembler::ClzR2(Register rd, Register rs) {
331 CHECK(!IsR6());
332 EmitR(0x1C, rs, rd, rd, 0, 0x20);
333}
334
335void MipsAssembler::CloR6(Register rd, Register rs) {
336 CHECK(IsR6());
337 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x11);
338}
339
340void MipsAssembler::CloR2(Register rd, Register rs) {
341 CHECK(!IsR6());
342 EmitR(0x1C, rs, rd, rd, 0, 0x21);
343}
344
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200345void MipsAssembler::Seb(Register rd, Register rt) {
346 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x10, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700347}
348
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200349void MipsAssembler::Seh(Register rd, Register rt) {
350 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x18, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700351}
352
Chris Larsen3f8bf652015-10-28 10:08:56 -0700353void MipsAssembler::Wsbh(Register rd, Register rt) {
354 EmitR(0x1f, static_cast<Register>(0), rt, rd, 2, 0x20);
355}
356
Chris Larsen70014c82015-11-18 12:26:08 -0800357void MipsAssembler::Bitswap(Register rd, Register rt) {
358 CHECK(IsR6());
359 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x0, 0x20);
360}
361
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200362void MipsAssembler::Sll(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700363 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200364 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x00);
jeffhao7fbee072012-08-24 17:56:54 -0700365}
366
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200367void MipsAssembler::Srl(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, 0x02);
370}
371
Chris Larsen3f8bf652015-10-28 10:08:56 -0700372void MipsAssembler::Rotr(Register rd, Register rt, int shamt) {
373 CHECK(IsUint<5>(shamt)) << shamt;
374 EmitR(0, static_cast<Register>(1), rt, rd, shamt, 0x02);
375}
376
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200377void MipsAssembler::Sra(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700378 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200379 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x03);
380}
381
382void MipsAssembler::Sllv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700383 EmitR(0, rs, rt, rd, 0, 0x04);
384}
385
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200386void MipsAssembler::Srlv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700387 EmitR(0, rs, rt, rd, 0, 0x06);
388}
389
Chris Larsene16ce5a2015-11-18 12:30:20 -0800390void MipsAssembler::Rotrv(Register rd, Register rt, Register rs) {
391 EmitR(0, rs, rt, rd, 1, 0x06);
392}
393
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200394void MipsAssembler::Srav(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700395 EmitR(0, rs, rt, rd, 0, 0x07);
396}
397
398void MipsAssembler::Lb(Register rt, Register rs, uint16_t imm16) {
399 EmitI(0x20, rs, rt, imm16);
400}
401
402void MipsAssembler::Lh(Register rt, Register rs, uint16_t imm16) {
403 EmitI(0x21, rs, rt, imm16);
404}
405
406void MipsAssembler::Lw(Register rt, Register rs, uint16_t imm16) {
407 EmitI(0x23, rs, rt, imm16);
408}
409
410void MipsAssembler::Lbu(Register rt, Register rs, uint16_t imm16) {
411 EmitI(0x24, rs, rt, imm16);
412}
413
414void MipsAssembler::Lhu(Register rt, Register rs, uint16_t imm16) {
415 EmitI(0x25, rs, rt, imm16);
416}
417
418void MipsAssembler::Lui(Register rt, uint16_t imm16) {
419 EmitI(0xf, static_cast<Register>(0), rt, imm16);
420}
421
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200422void MipsAssembler::Sync(uint32_t stype) {
423 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0),
424 stype & 0x1f, 0xf);
425}
426
jeffhao7fbee072012-08-24 17:56:54 -0700427void MipsAssembler::Mfhi(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200428 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700429 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x10);
430}
431
432void MipsAssembler::Mflo(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, 0x12);
435}
436
437void MipsAssembler::Sb(Register rt, Register rs, uint16_t imm16) {
438 EmitI(0x28, rs, rt, imm16);
439}
440
441void MipsAssembler::Sh(Register rt, Register rs, uint16_t imm16) {
442 EmitI(0x29, rs, rt, imm16);
443}
444
445void MipsAssembler::Sw(Register rt, Register rs, uint16_t imm16) {
446 EmitI(0x2b, rs, rt, imm16);
447}
448
449void MipsAssembler::Slt(Register rd, Register rs, Register rt) {
450 EmitR(0, rs, rt, rd, 0, 0x2a);
451}
452
453void MipsAssembler::Sltu(Register rd, Register rs, Register rt) {
454 EmitR(0, rs, rt, rd, 0, 0x2b);
455}
456
457void MipsAssembler::Slti(Register rt, Register rs, uint16_t imm16) {
458 EmitI(0xa, rs, rt, imm16);
459}
460
461void MipsAssembler::Sltiu(Register rt, Register rs, uint16_t imm16) {
462 EmitI(0xb, rs, rt, imm16);
463}
464
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200465void MipsAssembler::B(uint16_t imm16) {
466 EmitI(0x4, static_cast<Register>(0), static_cast<Register>(0), imm16);
467}
468
469void MipsAssembler::Beq(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700470 EmitI(0x4, rs, rt, imm16);
471}
472
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200473void MipsAssembler::Bne(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700474 EmitI(0x5, rs, rt, imm16);
475}
476
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200477void MipsAssembler::Beqz(Register rt, uint16_t imm16) {
478 Beq(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700479}
480
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200481void MipsAssembler::Bnez(Register rt, uint16_t imm16) {
482 Bne(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700483}
484
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200485void MipsAssembler::Bltz(Register rt, uint16_t imm16) {
486 EmitI(0x1, rt, static_cast<Register>(0), imm16);
487}
488
489void MipsAssembler::Bgez(Register rt, uint16_t imm16) {
490 EmitI(0x1, rt, static_cast<Register>(0x1), imm16);
491}
492
493void MipsAssembler::Blez(Register rt, uint16_t imm16) {
494 EmitI(0x6, rt, static_cast<Register>(0), imm16);
495}
496
497void MipsAssembler::Bgtz(Register rt, uint16_t imm16) {
498 EmitI(0x7, rt, static_cast<Register>(0), imm16);
499}
500
501void MipsAssembler::J(uint32_t addr26) {
502 EmitI26(0x2, addr26);
503}
504
505void MipsAssembler::Jal(uint32_t addr26) {
506 EmitI26(0x3, addr26);
507}
508
509void MipsAssembler::Jalr(Register rd, Register rs) {
510 EmitR(0, rs, static_cast<Register>(0), rd, 0, 0x09);
jeffhao7fbee072012-08-24 17:56:54 -0700511}
512
513void MipsAssembler::Jalr(Register rs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200514 Jalr(RA, rs);
515}
516
517void MipsAssembler::Jr(Register rs) {
518 Jalr(ZERO, rs);
519}
520
521void MipsAssembler::Nal() {
522 EmitI(0x1, static_cast<Register>(0), static_cast<Register>(0x10), 0);
523}
524
525void MipsAssembler::Auipc(Register rs, uint16_t imm16) {
526 CHECK(IsR6());
527 EmitI(0x3B, rs, static_cast<Register>(0x1E), imm16);
528}
529
530void MipsAssembler::Addiupc(Register rs, uint32_t imm19) {
531 CHECK(IsR6());
532 CHECK(IsUint<19>(imm19)) << imm19;
533 EmitI21(0x3B, rs, imm19);
534}
535
536void MipsAssembler::Bc(uint32_t imm26) {
537 CHECK(IsR6());
538 EmitI26(0x32, imm26);
539}
540
541void MipsAssembler::Jic(Register rt, uint16_t imm16) {
542 CHECK(IsR6());
543 EmitI(0x36, static_cast<Register>(0), rt, imm16);
544}
545
546void MipsAssembler::Jialc(Register rt, uint16_t imm16) {
547 CHECK(IsR6());
548 EmitI(0x3E, static_cast<Register>(0), rt, imm16);
549}
550
551void MipsAssembler::Bltc(Register rs, Register rt, uint16_t imm16) {
552 CHECK(IsR6());
553 CHECK_NE(rs, ZERO);
554 CHECK_NE(rt, ZERO);
555 CHECK_NE(rs, rt);
556 EmitI(0x17, rs, rt, imm16);
557}
558
559void MipsAssembler::Bltzc(Register rt, uint16_t imm16) {
560 CHECK(IsR6());
561 CHECK_NE(rt, ZERO);
562 EmitI(0x17, rt, rt, imm16);
563}
564
565void MipsAssembler::Bgtzc(Register rt, uint16_t imm16) {
566 CHECK(IsR6());
567 CHECK_NE(rt, ZERO);
568 EmitI(0x17, static_cast<Register>(0), rt, imm16);
569}
570
571void MipsAssembler::Bgec(Register rs, Register rt, uint16_t imm16) {
572 CHECK(IsR6());
573 CHECK_NE(rs, ZERO);
574 CHECK_NE(rt, ZERO);
575 CHECK_NE(rs, rt);
576 EmitI(0x16, rs, rt, imm16);
577}
578
579void MipsAssembler::Bgezc(Register rt, uint16_t imm16) {
580 CHECK(IsR6());
581 CHECK_NE(rt, ZERO);
582 EmitI(0x16, rt, rt, imm16);
583}
584
585void MipsAssembler::Blezc(Register rt, uint16_t imm16) {
586 CHECK(IsR6());
587 CHECK_NE(rt, ZERO);
588 EmitI(0x16, static_cast<Register>(0), rt, imm16);
589}
590
591void MipsAssembler::Bltuc(Register rs, Register rt, uint16_t imm16) {
592 CHECK(IsR6());
593 CHECK_NE(rs, ZERO);
594 CHECK_NE(rt, ZERO);
595 CHECK_NE(rs, rt);
596 EmitI(0x7, rs, rt, imm16);
597}
598
599void MipsAssembler::Bgeuc(Register rs, Register rt, uint16_t imm16) {
600 CHECK(IsR6());
601 CHECK_NE(rs, ZERO);
602 CHECK_NE(rt, ZERO);
603 CHECK_NE(rs, rt);
604 EmitI(0x6, rs, rt, imm16);
605}
606
607void MipsAssembler::Beqc(Register rs, Register rt, uint16_t imm16) {
608 CHECK(IsR6());
609 CHECK_NE(rs, ZERO);
610 CHECK_NE(rt, ZERO);
611 CHECK_NE(rs, rt);
612 EmitI(0x8, std::min(rs, rt), std::max(rs, rt), imm16);
613}
614
615void MipsAssembler::Bnec(Register rs, Register rt, uint16_t imm16) {
616 CHECK(IsR6());
617 CHECK_NE(rs, ZERO);
618 CHECK_NE(rt, ZERO);
619 CHECK_NE(rs, rt);
620 EmitI(0x18, std::min(rs, rt), std::max(rs, rt), imm16);
621}
622
623void MipsAssembler::Beqzc(Register rs, uint32_t imm21) {
624 CHECK(IsR6());
625 CHECK_NE(rs, ZERO);
626 EmitI21(0x36, rs, imm21);
627}
628
629void MipsAssembler::Bnezc(Register rs, uint32_t imm21) {
630 CHECK(IsR6());
631 CHECK_NE(rs, ZERO);
632 EmitI21(0x3E, rs, imm21);
633}
634
635void MipsAssembler::EmitBcond(BranchCondition cond, Register rs, Register rt, uint16_t imm16) {
636 switch (cond) {
637 case kCondLTZ:
638 CHECK_EQ(rt, ZERO);
639 Bltz(rs, imm16);
640 break;
641 case kCondGEZ:
642 CHECK_EQ(rt, ZERO);
643 Bgez(rs, imm16);
644 break;
645 case kCondLEZ:
646 CHECK_EQ(rt, ZERO);
647 Blez(rs, imm16);
648 break;
649 case kCondGTZ:
650 CHECK_EQ(rt, ZERO);
651 Bgtz(rs, imm16);
652 break;
653 case kCondEQ:
654 Beq(rs, rt, imm16);
655 break;
656 case kCondNE:
657 Bne(rs, rt, imm16);
658 break;
659 case kCondEQZ:
660 CHECK_EQ(rt, ZERO);
661 Beqz(rs, imm16);
662 break;
663 case kCondNEZ:
664 CHECK_EQ(rt, ZERO);
665 Bnez(rs, imm16);
666 break;
667 case kCondLT:
668 case kCondGE:
669 case kCondLE:
670 case kCondGT:
671 case kCondLTU:
672 case kCondGEU:
673 case kUncond:
674 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
675 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
676 LOG(FATAL) << "Unexpected branch condition " << cond;
677 UNREACHABLE();
678 }
679}
680
681void MipsAssembler::EmitBcondc(BranchCondition cond, Register rs, Register rt, uint32_t imm16_21) {
682 switch (cond) {
683 case kCondLT:
684 Bltc(rs, rt, imm16_21);
685 break;
686 case kCondGE:
687 Bgec(rs, rt, imm16_21);
688 break;
689 case kCondLE:
690 Bgec(rt, rs, imm16_21);
691 break;
692 case kCondGT:
693 Bltc(rt, rs, imm16_21);
694 break;
695 case kCondLTZ:
696 CHECK_EQ(rt, ZERO);
697 Bltzc(rs, imm16_21);
698 break;
699 case kCondGEZ:
700 CHECK_EQ(rt, ZERO);
701 Bgezc(rs, imm16_21);
702 break;
703 case kCondLEZ:
704 CHECK_EQ(rt, ZERO);
705 Blezc(rs, imm16_21);
706 break;
707 case kCondGTZ:
708 CHECK_EQ(rt, ZERO);
709 Bgtzc(rs, imm16_21);
710 break;
711 case kCondEQ:
712 Beqc(rs, rt, imm16_21);
713 break;
714 case kCondNE:
715 Bnec(rs, rt, imm16_21);
716 break;
717 case kCondEQZ:
718 CHECK_EQ(rt, ZERO);
719 Beqzc(rs, imm16_21);
720 break;
721 case kCondNEZ:
722 CHECK_EQ(rt, ZERO);
723 Bnezc(rs, imm16_21);
724 break;
725 case kCondLTU:
726 Bltuc(rs, rt, imm16_21);
727 break;
728 case kCondGEU:
729 Bgeuc(rs, rt, imm16_21);
730 break;
731 case kUncond:
732 LOG(FATAL) << "Unexpected branch condition " << cond;
733 UNREACHABLE();
734 }
jeffhao7fbee072012-08-24 17:56:54 -0700735}
736
737void MipsAssembler::AddS(FRegister fd, FRegister fs, FRegister ft) {
738 EmitFR(0x11, 0x10, ft, fs, fd, 0x0);
739}
740
741void MipsAssembler::SubS(FRegister fd, FRegister fs, FRegister ft) {
742 EmitFR(0x11, 0x10, ft, fs, fd, 0x1);
743}
744
745void MipsAssembler::MulS(FRegister fd, FRegister fs, FRegister ft) {
746 EmitFR(0x11, 0x10, ft, fs, fd, 0x2);
747}
748
749void MipsAssembler::DivS(FRegister fd, FRegister fs, FRegister ft) {
750 EmitFR(0x11, 0x10, ft, fs, fd, 0x3);
751}
752
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200753void MipsAssembler::AddD(FRegister fd, FRegister fs, FRegister ft) {
754 EmitFR(0x11, 0x11, ft, fs, fd, 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700755}
756
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200757void MipsAssembler::SubD(FRegister fd, FRegister fs, FRegister ft) {
758 EmitFR(0x11, 0x11, ft, fs, fd, 0x1);
jeffhao7fbee072012-08-24 17:56:54 -0700759}
760
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200761void MipsAssembler::MulD(FRegister fd, FRegister fs, FRegister ft) {
762 EmitFR(0x11, 0x11, ft, fs, fd, 0x2);
jeffhao7fbee072012-08-24 17:56:54 -0700763}
764
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200765void MipsAssembler::DivD(FRegister fd, FRegister fs, FRegister ft) {
766 EmitFR(0x11, 0x11, ft, fs, fd, 0x3);
jeffhao7fbee072012-08-24 17:56:54 -0700767}
768
769void MipsAssembler::MovS(FRegister fd, FRegister fs) {
770 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x6);
771}
772
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200773void MipsAssembler::MovD(FRegister fd, FRegister fs) {
774 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x6);
775}
776
777void MipsAssembler::NegS(FRegister fd, FRegister fs) {
778 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x7);
779}
780
781void MipsAssembler::NegD(FRegister fd, FRegister fs) {
782 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x7);
783}
784
785void MipsAssembler::Cvtsw(FRegister fd, FRegister fs) {
786 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x20);
787}
788
789void MipsAssembler::Cvtdw(FRegister fd, FRegister fs) {
790 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x21);
791}
792
793void MipsAssembler::Cvtsd(FRegister fd, FRegister fs) {
794 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x20);
795}
796
797void MipsAssembler::Cvtds(FRegister fd, FRegister fs) {
798 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x21);
jeffhao7fbee072012-08-24 17:56:54 -0700799}
800
801void MipsAssembler::Mfc1(Register rt, FRegister fs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200802 EmitFR(0x11, 0x00, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700803}
804
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200805void MipsAssembler::Mtc1(Register rt, FRegister fs) {
806 EmitFR(0x11, 0x04, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
807}
808
809void MipsAssembler::Mfhc1(Register rt, FRegister fs) {
810 EmitFR(0x11, 0x03, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
811}
812
813void MipsAssembler::Mthc1(Register rt, FRegister fs) {
814 EmitFR(0x11, 0x07, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700815}
816
817void MipsAssembler::Lwc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200818 EmitI(0x31, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700819}
820
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200821void MipsAssembler::Ldc1(FRegister ft, Register rs, uint16_t imm16) {
822 EmitI(0x35, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700823}
824
825void MipsAssembler::Swc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200826 EmitI(0x39, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700827}
828
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200829void MipsAssembler::Sdc1(FRegister ft, Register rs, uint16_t imm16) {
830 EmitI(0x3d, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700831}
832
833void MipsAssembler::Break() {
834 EmitR(0, static_cast<Register>(0), static_cast<Register>(0),
835 static_cast<Register>(0), 0, 0xD);
836}
837
jeffhao07030602012-09-26 14:33:14 -0700838void MipsAssembler::Nop() {
839 EmitR(0x0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0), 0, 0x0);
840}
841
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200842void MipsAssembler::Move(Register rd, Register rs) {
843 Or(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700844}
845
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200846void MipsAssembler::Clear(Register rd) {
847 Move(rd, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700848}
849
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200850void MipsAssembler::Not(Register rd, Register rs) {
851 Nor(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -0700852}
853
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200854void MipsAssembler::Push(Register rs) {
855 IncreaseFrameSize(kMipsWordSize);
856 Sw(rs, SP, 0);
jeffhao7fbee072012-08-24 17:56:54 -0700857}
858
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200859void MipsAssembler::Pop(Register rd) {
860 Lw(rd, SP, 0);
861 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -0700862}
863
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200864void MipsAssembler::PopAndReturn(Register rd, Register rt) {
865 Lw(rd, SP, 0);
866 Jr(rt);
867 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -0700868}
869
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200870void MipsAssembler::LoadConst32(Register rd, int32_t value) {
871 if (IsUint<16>(value)) {
872 // Use OR with (unsigned) immediate to encode 16b unsigned int.
873 Ori(rd, ZERO, value);
874 } else if (IsInt<16>(value)) {
875 // Use ADD with (signed) immediate to encode 16b signed int.
876 Addiu(rd, ZERO, value);
jeffhao7fbee072012-08-24 17:56:54 -0700877 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200878 Lui(rd, High16Bits(value));
879 if (value & 0xFFFF)
880 Ori(rd, rd, Low16Bits(value));
881 }
882}
883
884void MipsAssembler::LoadConst64(Register reg_hi, Register reg_lo, int64_t value) {
885 LoadConst32(reg_lo, Low32Bits(value));
886 LoadConst32(reg_hi, High32Bits(value));
887}
888
889void MipsAssembler::StoreConst32ToOffset(int32_t value,
890 Register base,
891 int32_t offset,
892 Register temp) {
893 if (!IsInt<16>(offset)) {
894 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
895 LoadConst32(AT, offset);
896 Addu(AT, AT, base);
897 base = AT;
898 offset = 0;
899 }
900 LoadConst32(temp, value);
901 Sw(temp, base, offset);
902}
903
904void MipsAssembler::StoreConst64ToOffset(int64_t value,
905 Register base,
906 int32_t offset,
907 Register temp) {
908 // IsInt<16> must be passed a signed value.
909 if (!IsInt<16>(offset) || !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize))) {
910 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
911 LoadConst32(AT, offset);
912 Addu(AT, AT, base);
913 base = AT;
914 offset = 0;
915 }
916 LoadConst32(temp, Low32Bits(value));
917 Sw(temp, base, offset);
918 LoadConst32(temp, High32Bits(value));
919 Sw(temp, base, offset + kMipsWordSize);
920}
921
922void MipsAssembler::LoadSConst32(FRegister r, int32_t value, Register temp) {
923 LoadConst32(temp, value);
924 Mtc1(temp, r);
925}
926
927void MipsAssembler::LoadDConst64(FRegister rd, int64_t value, Register temp) {
928 LoadConst32(temp, Low32Bits(value));
929 Mtc1(temp, rd);
930 LoadConst32(temp, High32Bits(value));
931 Mthc1(temp, rd);
932}
933
934void MipsAssembler::Addiu32(Register rt, Register rs, int32_t value, Register temp) {
935 if (IsInt<16>(value)) {
936 Addiu(rt, rs, value);
937 } else {
938 LoadConst32(temp, value);
939 Addu(rt, rs, temp);
940 }
941}
942
943void MipsAssembler::Branch::InitShortOrLong(MipsAssembler::Branch::OffsetBits offset_size,
944 MipsAssembler::Branch::Type short_type,
945 MipsAssembler::Branch::Type long_type) {
946 type_ = (offset_size <= branch_info_[short_type].offset_size) ? short_type : long_type;
947}
948
949void MipsAssembler::Branch::InitializeType(bool is_call, bool is_r6) {
950 OffsetBits offset_size = GetOffsetSizeNeeded(location_, target_);
951 if (is_r6) {
952 // R6
953 if (is_call) {
954 InitShortOrLong(offset_size, kR6Call, kR6LongCall);
955 } else if (condition_ == kUncond) {
956 InitShortOrLong(offset_size, kR6UncondBranch, kR6LongUncondBranch);
957 } else {
958 if (condition_ == kCondEQZ || condition_ == kCondNEZ) {
959 // Special case for beqzc/bnezc with longer offset than in other b<cond>c instructions.
960 type_ = (offset_size <= kOffset23) ? kR6CondBranch : kR6LongCondBranch;
961 } else {
962 InitShortOrLong(offset_size, kR6CondBranch, kR6LongCondBranch);
963 }
964 }
965 } else {
966 // R2
967 if (is_call) {
968 InitShortOrLong(offset_size, kCall, kLongCall);
969 } else if (condition_ == kUncond) {
970 InitShortOrLong(offset_size, kUncondBranch, kLongUncondBranch);
971 } else {
972 InitShortOrLong(offset_size, kCondBranch, kLongCondBranch);
973 }
974 }
975 old_type_ = type_;
976}
977
978bool MipsAssembler::Branch::IsNop(BranchCondition condition, Register lhs, Register rhs) {
979 switch (condition) {
980 case kCondLT:
981 case kCondGT:
982 case kCondNE:
983 case kCondLTU:
984 return lhs == rhs;
985 default:
986 return false;
987 }
988}
989
990bool MipsAssembler::Branch::IsUncond(BranchCondition condition, Register lhs, Register rhs) {
991 switch (condition) {
992 case kUncond:
993 return true;
994 case kCondGE:
995 case kCondLE:
996 case kCondEQ:
997 case kCondGEU:
998 return lhs == rhs;
999 default:
1000 return false;
1001 }
1002}
1003
1004MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target)
1005 : old_location_(location),
1006 location_(location),
1007 target_(target),
1008 lhs_reg_(0),
1009 rhs_reg_(0),
1010 condition_(kUncond) {
1011 InitializeType(false, is_r6);
1012}
1013
1014MipsAssembler::Branch::Branch(bool is_r6,
1015 uint32_t location,
1016 uint32_t target,
1017 MipsAssembler::BranchCondition condition,
1018 Register lhs_reg,
1019 Register rhs_reg)
1020 : old_location_(location),
1021 location_(location),
1022 target_(target),
1023 lhs_reg_(lhs_reg),
1024 rhs_reg_(rhs_reg),
1025 condition_(condition) {
1026 CHECK_NE(condition, kUncond);
1027 switch (condition) {
1028 case kCondLT:
1029 case kCondGE:
1030 case kCondLE:
1031 case kCondGT:
1032 case kCondLTU:
1033 case kCondGEU:
1034 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
1035 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
1036 // We leave this up to the caller.
1037 CHECK(is_r6);
1038 FALLTHROUGH_INTENDED;
1039 case kCondEQ:
1040 case kCondNE:
1041 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1042 // To compare with 0, use dedicated kCond*Z conditions.
1043 CHECK_NE(lhs_reg, ZERO);
1044 CHECK_NE(rhs_reg, ZERO);
1045 break;
1046 case kCondLTZ:
1047 case kCondGEZ:
1048 case kCondLEZ:
1049 case kCondGTZ:
1050 case kCondEQZ:
1051 case kCondNEZ:
1052 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1053 CHECK_NE(lhs_reg, ZERO);
1054 CHECK_EQ(rhs_reg, ZERO);
1055 break;
1056 case kUncond:
1057 UNREACHABLE();
1058 }
1059 CHECK(!IsNop(condition, lhs_reg, rhs_reg));
1060 if (IsUncond(condition, lhs_reg, rhs_reg)) {
1061 // Branch condition is always true, make the branch unconditional.
1062 condition_ = kUncond;
1063 }
1064 InitializeType(false, is_r6);
1065}
1066
1067MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target, Register indirect_reg)
1068 : old_location_(location),
1069 location_(location),
1070 target_(target),
1071 lhs_reg_(indirect_reg),
1072 rhs_reg_(0),
1073 condition_(kUncond) {
1074 CHECK_NE(indirect_reg, ZERO);
1075 CHECK_NE(indirect_reg, AT);
1076 InitializeType(true, is_r6);
1077}
1078
1079MipsAssembler::BranchCondition MipsAssembler::Branch::OppositeCondition(
1080 MipsAssembler::BranchCondition cond) {
1081 switch (cond) {
1082 case kCondLT:
1083 return kCondGE;
1084 case kCondGE:
1085 return kCondLT;
1086 case kCondLE:
1087 return kCondGT;
1088 case kCondGT:
1089 return kCondLE;
1090 case kCondLTZ:
1091 return kCondGEZ;
1092 case kCondGEZ:
1093 return kCondLTZ;
1094 case kCondLEZ:
1095 return kCondGTZ;
1096 case kCondGTZ:
1097 return kCondLEZ;
1098 case kCondEQ:
1099 return kCondNE;
1100 case kCondNE:
1101 return kCondEQ;
1102 case kCondEQZ:
1103 return kCondNEZ;
1104 case kCondNEZ:
1105 return kCondEQZ;
1106 case kCondLTU:
1107 return kCondGEU;
1108 case kCondGEU:
1109 return kCondLTU;
1110 case kUncond:
1111 LOG(FATAL) << "Unexpected branch condition " << cond;
1112 }
1113 UNREACHABLE();
1114}
1115
1116MipsAssembler::Branch::Type MipsAssembler::Branch::GetType() const {
1117 return type_;
1118}
1119
1120MipsAssembler::BranchCondition MipsAssembler::Branch::GetCondition() const {
1121 return condition_;
1122}
1123
1124Register MipsAssembler::Branch::GetLeftRegister() const {
1125 return static_cast<Register>(lhs_reg_);
1126}
1127
1128Register MipsAssembler::Branch::GetRightRegister() const {
1129 return static_cast<Register>(rhs_reg_);
1130}
1131
1132uint32_t MipsAssembler::Branch::GetTarget() const {
1133 return target_;
1134}
1135
1136uint32_t MipsAssembler::Branch::GetLocation() const {
1137 return location_;
1138}
1139
1140uint32_t MipsAssembler::Branch::GetOldLocation() const {
1141 return old_location_;
1142}
1143
1144uint32_t MipsAssembler::Branch::GetLength() const {
1145 return branch_info_[type_].length;
1146}
1147
1148uint32_t MipsAssembler::Branch::GetOldLength() const {
1149 return branch_info_[old_type_].length;
1150}
1151
1152uint32_t MipsAssembler::Branch::GetSize() const {
1153 return GetLength() * sizeof(uint32_t);
1154}
1155
1156uint32_t MipsAssembler::Branch::GetOldSize() const {
1157 return GetOldLength() * sizeof(uint32_t);
1158}
1159
1160uint32_t MipsAssembler::Branch::GetEndLocation() const {
1161 return GetLocation() + GetSize();
1162}
1163
1164uint32_t MipsAssembler::Branch::GetOldEndLocation() const {
1165 return GetOldLocation() + GetOldSize();
1166}
1167
1168bool MipsAssembler::Branch::IsLong() const {
1169 switch (type_) {
1170 // R2 short branches.
1171 case kUncondBranch:
1172 case kCondBranch:
1173 case kCall:
1174 // R6 short branches.
1175 case kR6UncondBranch:
1176 case kR6CondBranch:
1177 case kR6Call:
1178 return false;
1179 // R2 long branches.
1180 case kLongUncondBranch:
1181 case kLongCondBranch:
1182 case kLongCall:
1183 // R6 long branches.
1184 case kR6LongUncondBranch:
1185 case kR6LongCondBranch:
1186 case kR6LongCall:
1187 return true;
1188 }
1189 UNREACHABLE();
1190}
1191
1192bool MipsAssembler::Branch::IsResolved() const {
1193 return target_ != kUnresolved;
1194}
1195
1196MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSize() const {
1197 OffsetBits offset_size =
1198 (type_ == kR6CondBranch && (condition_ == kCondEQZ || condition_ == kCondNEZ))
1199 ? kOffset23
1200 : branch_info_[type_].offset_size;
1201 return offset_size;
1202}
1203
1204MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSizeNeeded(uint32_t location,
1205 uint32_t target) {
1206 // For unresolved targets assume the shortest encoding
1207 // (later it will be made longer if needed).
1208 if (target == kUnresolved)
1209 return kOffset16;
1210 int64_t distance = static_cast<int64_t>(target) - location;
1211 // To simplify calculations in composite branches consisting of multiple instructions
1212 // bump up the distance by a value larger than the max byte size of a composite branch.
1213 distance += (distance >= 0) ? kMaxBranchSize : -kMaxBranchSize;
1214 if (IsInt<kOffset16>(distance))
1215 return kOffset16;
1216 else if (IsInt<kOffset18>(distance))
1217 return kOffset18;
1218 else if (IsInt<kOffset21>(distance))
1219 return kOffset21;
1220 else if (IsInt<kOffset23>(distance))
1221 return kOffset23;
1222 else if (IsInt<kOffset28>(distance))
1223 return kOffset28;
1224 return kOffset32;
1225}
1226
1227void MipsAssembler::Branch::Resolve(uint32_t target) {
1228 target_ = target;
1229}
1230
1231void MipsAssembler::Branch::Relocate(uint32_t expand_location, uint32_t delta) {
1232 if (location_ > expand_location) {
1233 location_ += delta;
1234 }
1235 if (!IsResolved()) {
1236 return; // Don't know the target yet.
1237 }
1238 if (target_ > expand_location) {
1239 target_ += delta;
1240 }
1241}
1242
1243void MipsAssembler::Branch::PromoteToLong() {
1244 switch (type_) {
1245 // R2 short branches.
1246 case kUncondBranch:
1247 type_ = kLongUncondBranch;
1248 break;
1249 case kCondBranch:
1250 type_ = kLongCondBranch;
1251 break;
1252 case kCall:
1253 type_ = kLongCall;
1254 break;
1255 // R6 short branches.
1256 case kR6UncondBranch:
1257 type_ = kR6LongUncondBranch;
1258 break;
1259 case kR6CondBranch:
1260 type_ = kR6LongCondBranch;
1261 break;
1262 case kR6Call:
1263 type_ = kR6LongCall;
1264 break;
1265 default:
1266 // Note: 'type_' is already long.
1267 break;
1268 }
1269 CHECK(IsLong());
1270}
1271
1272uint32_t MipsAssembler::Branch::PromoteIfNeeded(uint32_t max_short_distance) {
1273 // If the branch is still unresolved or already long, nothing to do.
1274 if (IsLong() || !IsResolved()) {
1275 return 0;
1276 }
1277 // Promote the short branch to long if the offset size is too small
1278 // to hold the distance between location_ and target_.
1279 if (GetOffsetSizeNeeded(location_, target_) > GetOffsetSize()) {
1280 PromoteToLong();
1281 uint32_t old_size = GetOldSize();
1282 uint32_t new_size = GetSize();
1283 CHECK_GT(new_size, old_size);
1284 return new_size - old_size;
1285 }
1286 // The following logic is for debugging/testing purposes.
1287 // Promote some short branches to long when it's not really required.
1288 if (UNLIKELY(max_short_distance != std::numeric_limits<uint32_t>::max())) {
1289 int64_t distance = static_cast<int64_t>(target_) - location_;
1290 distance = (distance >= 0) ? distance : -distance;
1291 if (distance >= max_short_distance) {
1292 PromoteToLong();
1293 uint32_t old_size = GetOldSize();
1294 uint32_t new_size = GetSize();
1295 CHECK_GT(new_size, old_size);
1296 return new_size - old_size;
1297 }
1298 }
1299 return 0;
1300}
1301
1302uint32_t MipsAssembler::Branch::GetOffsetLocation() const {
1303 return location_ + branch_info_[type_].instr_offset * sizeof(uint32_t);
1304}
1305
1306uint32_t MipsAssembler::Branch::GetOffset() const {
1307 CHECK(IsResolved());
1308 uint32_t ofs_mask = 0xFFFFFFFF >> (32 - GetOffsetSize());
1309 // Calculate the byte distance between instructions and also account for
1310 // different PC-relative origins.
1311 uint32_t offset = target_ - GetOffsetLocation() - branch_info_[type_].pc_org * sizeof(uint32_t);
1312 // Prepare the offset for encoding into the instruction(s).
1313 offset = (offset & ofs_mask) >> branch_info_[type_].offset_shift;
1314 return offset;
1315}
1316
1317MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) {
1318 CHECK_LT(branch_id, branches_.size());
1319 return &branches_[branch_id];
1320}
1321
1322const MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) const {
1323 CHECK_LT(branch_id, branches_.size());
1324 return &branches_[branch_id];
1325}
1326
1327void MipsAssembler::Bind(MipsLabel* label) {
1328 CHECK(!label->IsBound());
1329 uint32_t bound_pc = buffer_.Size();
1330
1331 // Walk the list of branches referring to and preceding this label.
1332 // Store the previously unknown target addresses in them.
1333 while (label->IsLinked()) {
1334 uint32_t branch_id = label->Position();
1335 Branch* branch = GetBranch(branch_id);
1336 branch->Resolve(bound_pc);
1337
1338 uint32_t branch_location = branch->GetLocation();
1339 // Extract the location of the previous branch in the list (walking the list backwards;
1340 // the previous branch ID was stored in the space reserved for this branch).
1341 uint32_t prev = buffer_.Load<uint32_t>(branch_location);
1342
1343 // On to the previous branch in the list...
1344 label->position_ = prev;
1345 }
1346
1347 // Now make the label object contain its own location (relative to the end of the preceding
1348 // branch, if any; it will be used by the branches referring to and following this label).
1349 label->prev_branch_id_plus_one_ = branches_.size();
1350 if (label->prev_branch_id_plus_one_) {
1351 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1352 const Branch* branch = GetBranch(branch_id);
1353 bound_pc -= branch->GetEndLocation();
1354 }
1355 label->BindTo(bound_pc);
1356}
1357
1358uint32_t MipsAssembler::GetLabelLocation(MipsLabel* label) const {
1359 CHECK(label->IsBound());
1360 uint32_t target = label->Position();
1361 if (label->prev_branch_id_plus_one_) {
1362 // Get label location based on the branch preceding it.
1363 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1364 const Branch* branch = GetBranch(branch_id);
1365 target += branch->GetEndLocation();
1366 }
1367 return target;
1368}
1369
1370uint32_t MipsAssembler::GetAdjustedPosition(uint32_t old_position) {
1371 // We can reconstruct the adjustment by going through all the branches from the beginning
1372 // up to the old_position. Since we expect AdjustedPosition() to be called in a loop
1373 // with increasing old_position, we can use the data from last AdjustedPosition() to
1374 // continue where we left off and the whole loop should be O(m+n) where m is the number
1375 // of positions to adjust and n is the number of branches.
1376 if (old_position < last_old_position_) {
1377 last_position_adjustment_ = 0;
1378 last_old_position_ = 0;
1379 last_branch_id_ = 0;
1380 }
1381 while (last_branch_id_ != branches_.size()) {
1382 const Branch* branch = GetBranch(last_branch_id_);
1383 if (branch->GetLocation() >= old_position + last_position_adjustment_) {
1384 break;
1385 }
1386 last_position_adjustment_ += branch->GetSize() - branch->GetOldSize();
1387 ++last_branch_id_;
1388 }
1389 last_old_position_ = old_position;
1390 return old_position + last_position_adjustment_;
1391}
1392
1393void MipsAssembler::FinalizeLabeledBranch(MipsLabel* label) {
1394 uint32_t length = branches_.back().GetLength();
1395 if (!label->IsBound()) {
1396 // Branch forward (to a following label), distance is unknown.
1397 // The first branch forward will contain 0, serving as the terminator of
1398 // the list of forward-reaching branches.
1399 Emit(label->position_);
1400 length--;
1401 // Now make the label object point to this branch
1402 // (this forms a linked list of branches preceding this label).
1403 uint32_t branch_id = branches_.size() - 1;
1404 label->LinkTo(branch_id);
1405 }
1406 // Reserve space for the branch.
1407 while (length--) {
1408 Nop();
1409 }
1410}
1411
1412void MipsAssembler::Buncond(MipsLabel* label) {
1413 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1414 branches_.emplace_back(IsR6(), buffer_.Size(), target);
1415 FinalizeLabeledBranch(label);
1416}
1417
1418void MipsAssembler::Bcond(MipsLabel* label, BranchCondition condition, Register lhs, Register rhs) {
1419 // If lhs = rhs, this can be a NOP.
1420 if (Branch::IsNop(condition, lhs, rhs)) {
1421 return;
1422 }
1423 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1424 branches_.emplace_back(IsR6(), buffer_.Size(), target, condition, lhs, rhs);
1425 FinalizeLabeledBranch(label);
1426}
1427
1428void MipsAssembler::Call(MipsLabel* label, Register indirect_reg) {
1429 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1430 branches_.emplace_back(IsR6(), buffer_.Size(), target, indirect_reg);
1431 FinalizeLabeledBranch(label);
1432}
1433
1434void MipsAssembler::PromoteBranches() {
1435 // Promote short branches to long as necessary.
1436 bool changed;
1437 do {
1438 changed = false;
1439 for (auto& branch : branches_) {
1440 CHECK(branch.IsResolved());
1441 uint32_t delta = branch.PromoteIfNeeded();
1442 // If this branch has been promoted and needs to expand in size,
1443 // relocate all branches by the expansion size.
1444 if (delta) {
1445 changed = true;
1446 uint32_t expand_location = branch.GetLocation();
1447 for (auto& branch2 : branches_) {
1448 branch2.Relocate(expand_location, delta);
1449 }
1450 }
1451 }
1452 } while (changed);
1453
1454 // Account for branch expansion by resizing the code buffer
1455 // and moving the code in it to its final location.
1456 size_t branch_count = branches_.size();
1457 if (branch_count > 0) {
1458 // Resize.
1459 Branch& last_branch = branches_[branch_count - 1];
1460 uint32_t size_delta = last_branch.GetEndLocation() - last_branch.GetOldEndLocation();
1461 uint32_t old_size = buffer_.Size();
1462 buffer_.Resize(old_size + size_delta);
1463 // Move the code residing between branch placeholders.
1464 uint32_t end = old_size;
1465 for (size_t i = branch_count; i > 0; ) {
1466 Branch& branch = branches_[--i];
1467 uint32_t size = end - branch.GetOldEndLocation();
1468 buffer_.Move(branch.GetEndLocation(), branch.GetOldEndLocation(), size);
1469 end = branch.GetOldLocation();
1470 }
1471 }
1472}
1473
1474// Note: make sure branch_info_[] and EmitBranch() are kept synchronized.
1475const MipsAssembler::Branch::BranchInfo MipsAssembler::Branch::branch_info_[] = {
1476 // R2 short branches.
1477 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kUncondBranch
1478 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kCondBranch
1479 { 5, 2, 0, MipsAssembler::Branch::kOffset16, 0 }, // kCall
1480 // R2 long branches.
1481 { 9, 3, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongUncondBranch
1482 { 10, 4, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCondBranch
1483 { 6, 1, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCall
1484 // R6 short branches.
1485 { 1, 0, 1, MipsAssembler::Branch::kOffset28, 2 }, // kR6UncondBranch
1486 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kR6CondBranch
1487 // Exception: kOffset23 for beqzc/bnezc.
1488 { 2, 0, 0, MipsAssembler::Branch::kOffset21, 2 }, // kR6Call
1489 // R6 long branches.
1490 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongUncondBranch
1491 { 3, 1, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCondBranch
1492 { 3, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCall
1493};
1494
1495// Note: make sure branch_info_[] and mitBranch() are kept synchronized.
1496void MipsAssembler::EmitBranch(MipsAssembler::Branch* branch) {
1497 CHECK_EQ(overwriting_, true);
1498 overwrite_location_ = branch->GetLocation();
1499 uint32_t offset = branch->GetOffset();
1500 BranchCondition condition = branch->GetCondition();
1501 Register lhs = branch->GetLeftRegister();
1502 Register rhs = branch->GetRightRegister();
1503 switch (branch->GetType()) {
1504 // R2 short branches.
1505 case Branch::kUncondBranch:
1506 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1507 B(offset);
1508 Nop(); // TODO: improve by filling the delay slot.
1509 break;
1510 case Branch::kCondBranch:
1511 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1512 EmitBcond(condition, lhs, rhs, offset);
1513 Nop(); // TODO: improve by filling the delay slot.
1514 break;
1515 case Branch::kCall:
1516 Nal();
1517 Nop(); // TODO: is this NOP really needed here?
1518 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1519 Addiu(lhs, RA, offset);
1520 Jalr(lhs);
1521 Nop();
1522 break;
1523
1524 // R2 long branches.
1525 case Branch::kLongUncondBranch:
1526 // To get the value of the PC register we need to use the NAL instruction.
1527 // NAL clobbers the RA register. However, RA must be preserved if the
1528 // method is compiled without the entry/exit sequences that would take care
1529 // of preserving RA (typically, leaf methods don't preserve RA explicitly).
1530 // So, we need to preserve RA in some temporary storage ourselves. The AT
1531 // register can't be used for this because we need it to load a constant
1532 // which will be added to the value that NAL stores in RA. And we can't
1533 // use T9 for this in the context of the JNI compiler, which uses it
1534 // as a scratch register (see InterproceduralScratchRegister()).
1535 // If we were to add a 32-bit constant to RA using two ADDIU instructions,
1536 // we'd also need to use the ROTR instruction, which requires no less than
1537 // MIPSR2.
1538 // Perhaps, we could use T8 or one of R2's multiplier/divider registers
1539 // (LO or HI) or even a floating-point register, but that doesn't seem
1540 // like a nice solution. We may want this to work on both R6 and pre-R6.
1541 // For now simply use the stack for RA. This should be OK since for the
1542 // vast majority of code a short PC-relative branch is sufficient.
1543 // TODO: can this be improved?
1544 Push(RA);
1545 Nal();
1546 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1547 Lui(AT, High16Bits(offset));
1548 Ori(AT, AT, Low16Bits(offset));
1549 Addu(AT, AT, RA);
1550 Lw(RA, SP, 0);
1551 Jr(AT);
1552 DecreaseFrameSize(kMipsWordSize);
1553 break;
1554 case Branch::kLongCondBranch:
1555 // The comment on case 'Branch::kLongUncondBranch' applies here as well.
1556 // Note: the opposite condition branch encodes 8 as the distance, which is equal to the
1557 // number of instructions skipped:
1558 // (PUSH(IncreaseFrameSize(ADDIU) + SW) + NAL + LUI + ORI + ADDU + LW + JR).
1559 EmitBcond(Branch::OppositeCondition(condition), lhs, rhs, 8);
1560 Push(RA);
1561 Nal();
1562 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1563 Lui(AT, High16Bits(offset));
1564 Ori(AT, AT, Low16Bits(offset));
1565 Addu(AT, AT, RA);
1566 Lw(RA, SP, 0);
1567 Jr(AT);
1568 DecreaseFrameSize(kMipsWordSize);
1569 break;
1570 case Branch::kLongCall:
1571 Nal();
1572 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1573 Lui(AT, High16Bits(offset));
1574 Ori(AT, AT, Low16Bits(offset));
1575 Addu(lhs, AT, RA);
1576 Jalr(lhs);
1577 Nop();
1578 break;
1579
1580 // R6 short branches.
1581 case Branch::kR6UncondBranch:
1582 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1583 Bc(offset);
1584 break;
1585 case Branch::kR6CondBranch:
1586 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1587 EmitBcondc(condition, lhs, rhs, offset);
1588 Nop(); // TODO: improve by filling the forbidden slot.
1589 break;
1590 case Branch::kR6Call:
1591 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1592 Addiupc(lhs, offset);
1593 Jialc(lhs, 0);
1594 break;
1595
1596 // R6 long branches.
1597 case Branch::kR6LongUncondBranch:
1598 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1599 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1600 Auipc(AT, High16Bits(offset));
1601 Jic(AT, Low16Bits(offset));
1602 break;
1603 case Branch::kR6LongCondBranch:
1604 EmitBcondc(Branch::OppositeCondition(condition), lhs, rhs, 2);
1605 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1606 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1607 Auipc(AT, High16Bits(offset));
1608 Jic(AT, Low16Bits(offset));
1609 break;
1610 case Branch::kR6LongCall:
1611 offset += (offset & 0x8000) << 1; // Account for sign extension in addiu.
1612 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1613 Auipc(lhs, High16Bits(offset));
1614 Addiu(lhs, lhs, Low16Bits(offset));
1615 Jialc(lhs, 0);
1616 break;
1617 }
1618 CHECK_EQ(overwrite_location_, branch->GetEndLocation());
1619 CHECK_LT(branch->GetSize(), static_cast<uint32_t>(Branch::kMaxBranchSize));
1620}
1621
1622void MipsAssembler::B(MipsLabel* label) {
1623 Buncond(label);
1624}
1625
1626void MipsAssembler::Jalr(MipsLabel* label, Register indirect_reg) {
1627 Call(label, indirect_reg);
1628}
1629
1630void MipsAssembler::Beq(Register rs, Register rt, MipsLabel* label) {
1631 Bcond(label, kCondEQ, rs, rt);
1632}
1633
1634void MipsAssembler::Bne(Register rs, Register rt, MipsLabel* label) {
1635 Bcond(label, kCondNE, rs, rt);
1636}
1637
1638void MipsAssembler::Beqz(Register rt, MipsLabel* label) {
1639 Bcond(label, kCondEQZ, rt);
1640}
1641
1642void MipsAssembler::Bnez(Register rt, MipsLabel* label) {
1643 Bcond(label, kCondNEZ, rt);
1644}
1645
1646void MipsAssembler::Bltz(Register rt, MipsLabel* label) {
1647 Bcond(label, kCondLTZ, rt);
1648}
1649
1650void MipsAssembler::Bgez(Register rt, MipsLabel* label) {
1651 Bcond(label, kCondGEZ, rt);
1652}
1653
1654void MipsAssembler::Blez(Register rt, MipsLabel* label) {
1655 Bcond(label, kCondLEZ, rt);
1656}
1657
1658void MipsAssembler::Bgtz(Register rt, MipsLabel* label) {
1659 Bcond(label, kCondGTZ, rt);
1660}
1661
1662void MipsAssembler::Blt(Register rs, Register rt, MipsLabel* label) {
1663 if (IsR6()) {
1664 Bcond(label, kCondLT, rs, rt);
1665 } else if (!Branch::IsNop(kCondLT, rs, rt)) {
1666 // Synthesize the instruction (not available on R2).
1667 Slt(AT, rs, rt);
1668 Bnez(AT, label);
1669 }
1670}
1671
1672void MipsAssembler::Bge(Register rs, Register rt, MipsLabel* label) {
1673 if (IsR6()) {
1674 Bcond(label, kCondGE, rs, rt);
1675 } else if (Branch::IsUncond(kCondGE, rs, rt)) {
1676 B(label);
1677 } else {
1678 // Synthesize the instruction (not available on R2).
1679 Slt(AT, rs, rt);
1680 Beqz(AT, label);
1681 }
1682}
1683
1684void MipsAssembler::Bltu(Register rs, Register rt, MipsLabel* label) {
1685 if (IsR6()) {
1686 Bcond(label, kCondLTU, rs, rt);
1687 } else if (!Branch::IsNop(kCondLTU, rs, rt)) {
1688 // Synthesize the instruction (not available on R2).
1689 Sltu(AT, rs, rt);
1690 Bnez(AT, label);
1691 }
1692}
1693
1694void MipsAssembler::Bgeu(Register rs, Register rt, MipsLabel* label) {
1695 if (IsR6()) {
1696 Bcond(label, kCondGEU, rs, rt);
1697 } else if (Branch::IsUncond(kCondGEU, rs, rt)) {
1698 B(label);
1699 } else {
1700 // Synthesize the instruction (not available on R2).
1701 Sltu(AT, rs, rt);
1702 Beqz(AT, label);
jeffhao7fbee072012-08-24 17:56:54 -07001703 }
1704}
1705
1706void MipsAssembler::LoadFromOffset(LoadOperandType type, Register reg, Register base,
1707 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001708 // IsInt<16> must be passed a signed value.
1709 if (!IsInt<16>(offset) ||
1710 (type == kLoadDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1711 LoadConst32(AT, offset);
1712 Addu(AT, AT, base);
1713 base = AT;
1714 offset = 0;
1715 }
1716
jeffhao7fbee072012-08-24 17:56:54 -07001717 switch (type) {
1718 case kLoadSignedByte:
1719 Lb(reg, base, offset);
1720 break;
1721 case kLoadUnsignedByte:
1722 Lbu(reg, base, offset);
1723 break;
1724 case kLoadSignedHalfword:
1725 Lh(reg, base, offset);
1726 break;
1727 case kLoadUnsignedHalfword:
1728 Lhu(reg, base, offset);
1729 break;
1730 case kLoadWord:
1731 Lw(reg, base, offset);
1732 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001733 case kLoadDoubleword:
1734 if (reg == base) {
1735 // This will clobber the base when loading the lower register. Since we have to load the
1736 // higher register as well, this will fail. Solution: reverse the order.
1737 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
1738 Lw(reg, base, offset);
1739 } else {
1740 Lw(reg, base, offset);
1741 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
1742 }
jeffhao7fbee072012-08-24 17:56:54 -07001743 break;
1744 default:
1745 LOG(FATAL) << "UNREACHABLE";
1746 }
1747}
1748
1749void MipsAssembler::LoadSFromOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001750 if (!IsInt<16>(offset)) {
1751 LoadConst32(AT, offset);
1752 Addu(AT, AT, base);
1753 base = AT;
1754 offset = 0;
1755 }
1756
jeffhao7fbee072012-08-24 17:56:54 -07001757 Lwc1(reg, base, offset);
1758}
1759
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001760void MipsAssembler::LoadDFromOffset(FRegister reg, Register base, int32_t offset) {
1761 // IsInt<16> must be passed a signed value.
1762 if (!IsInt<16>(offset) ||
1763 (!IsAligned<kMipsDoublewordSize>(offset) &&
1764 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1765 LoadConst32(AT, offset);
1766 Addu(AT, AT, base);
1767 base = AT;
1768 offset = 0;
1769 }
1770
1771 if (offset & 0x7) {
1772 if (Is32BitFPU()) {
1773 Lwc1(reg, base, offset);
1774 Lwc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
1775 } else {
1776 // 64-bit FPU.
1777 Lwc1(reg, base, offset);
1778 Lw(T8, base, offset + kMipsWordSize);
1779 Mthc1(T8, reg);
1780 }
1781 } else {
1782 Ldc1(reg, base, offset);
1783 }
1784}
1785
1786void MipsAssembler::EmitLoad(ManagedRegister m_dst, Register src_register, int32_t src_offset,
1787 size_t size) {
1788 MipsManagedRegister dst = m_dst.AsMips();
1789 if (dst.IsNoRegister()) {
1790 CHECK_EQ(0u, size) << dst;
1791 } else if (dst.IsCoreRegister()) {
1792 CHECK_EQ(kMipsWordSize, size) << dst;
1793 LoadFromOffset(kLoadWord, dst.AsCoreRegister(), src_register, src_offset);
1794 } else if (dst.IsRegisterPair()) {
1795 CHECK_EQ(kMipsDoublewordSize, size) << dst;
1796 LoadFromOffset(kLoadDoubleword, dst.AsRegisterPairLow(), src_register, src_offset);
1797 } else if (dst.IsFRegister()) {
1798 if (size == kMipsWordSize) {
1799 LoadSFromOffset(dst.AsFRegister(), src_register, src_offset);
1800 } else {
1801 CHECK_EQ(kMipsDoublewordSize, size) << dst;
1802 LoadDFromOffset(dst.AsFRegister(), src_register, src_offset);
1803 }
1804 }
jeffhao7fbee072012-08-24 17:56:54 -07001805}
1806
1807void MipsAssembler::StoreToOffset(StoreOperandType type, Register reg, Register base,
1808 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001809 // IsInt<16> must be passed a signed value.
1810 if (!IsInt<16>(offset) ||
1811 (type == kStoreDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1812 LoadConst32(AT, offset);
1813 Addu(AT, AT, base);
1814 base = AT;
1815 offset = 0;
1816 }
1817
jeffhao7fbee072012-08-24 17:56:54 -07001818 switch (type) {
1819 case kStoreByte:
1820 Sb(reg, base, offset);
1821 break;
1822 case kStoreHalfword:
1823 Sh(reg, base, offset);
1824 break;
1825 case kStoreWord:
1826 Sw(reg, base, offset);
1827 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001828 case kStoreDoubleword:
1829 CHECK_NE(reg, base);
1830 CHECK_NE(static_cast<Register>(reg + 1), base);
1831 Sw(reg, base, offset);
1832 Sw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001833 break;
1834 default:
1835 LOG(FATAL) << "UNREACHABLE";
1836 }
1837}
1838
Goran Jakovljevicff734982015-08-24 12:58:55 +00001839void MipsAssembler::StoreSToOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001840 if (!IsInt<16>(offset)) {
1841 LoadConst32(AT, offset);
1842 Addu(AT, AT, base);
1843 base = AT;
1844 offset = 0;
1845 }
1846
jeffhao7fbee072012-08-24 17:56:54 -07001847 Swc1(reg, base, offset);
1848}
1849
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001850void MipsAssembler::StoreDToOffset(FRegister reg, Register base, int32_t offset) {
1851 // IsInt<16> must be passed a signed value.
1852 if (!IsInt<16>(offset) ||
1853 (!IsAligned<kMipsDoublewordSize>(offset) &&
1854 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1855 LoadConst32(AT, offset);
1856 Addu(AT, AT, base);
1857 base = AT;
1858 offset = 0;
1859 }
1860
1861 if (offset & 0x7) {
1862 if (Is32BitFPU()) {
1863 Swc1(reg, base, offset);
1864 Swc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
1865 } else {
1866 // 64-bit FPU.
1867 Mfhc1(T8, reg);
1868 Swc1(reg, base, offset);
1869 Sw(T8, base, offset + kMipsWordSize);
1870 }
1871 } else {
1872 Sdc1(reg, base, offset);
1873 }
jeffhao7fbee072012-08-24 17:56:54 -07001874}
1875
David Srbeckydd973932015-04-07 20:29:48 +01001876static dwarf::Reg DWARFReg(Register reg) {
1877 return dwarf::Reg::MipsCore(static_cast<int>(reg));
1878}
1879
Ian Rogers790a6b72014-04-01 10:36:00 -07001880constexpr size_t kFramePointerSize = 4;
1881
jeffhao7fbee072012-08-24 17:56:54 -07001882void MipsAssembler::BuildFrame(size_t frame_size, ManagedRegister method_reg,
1883 const std::vector<ManagedRegister>& callee_save_regs,
Dmitry Petrochenkofca82202014-03-21 11:21:37 +07001884 const ManagedRegisterEntrySpills& entry_spills) {
jeffhao7fbee072012-08-24 17:56:54 -07001885 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001886 DCHECK(!overwriting_);
jeffhao7fbee072012-08-24 17:56:54 -07001887
1888 // Increase frame to required size.
1889 IncreaseFrameSize(frame_size);
1890
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001891 // Push callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07001892 int stack_offset = frame_size - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001893 StoreToOffset(kStoreWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001894 cfi_.RelOffset(DWARFReg(RA), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07001895 for (int i = callee_save_regs.size() - 1; i >= 0; --i) {
Ian Rogers790a6b72014-04-01 10:36:00 -07001896 stack_offset -= kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001897 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
1898 StoreToOffset(kStoreWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001899 cfi_.RelOffset(DWARFReg(reg), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07001900 }
1901
1902 // Write out Method*.
1903 StoreToOffset(kStoreWord, method_reg.AsMips().AsCoreRegister(), SP, 0);
1904
1905 // Write out entry spills.
Goran Jakovljevicff734982015-08-24 12:58:55 +00001906 int32_t offset = frame_size + kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001907 for (size_t i = 0; i < entry_spills.size(); ++i) {
Goran Jakovljevicff734982015-08-24 12:58:55 +00001908 MipsManagedRegister reg = entry_spills.at(i).AsMips();
1909 if (reg.IsNoRegister()) {
1910 ManagedRegisterSpill spill = entry_spills.at(i);
1911 offset += spill.getSize();
1912 } else if (reg.IsCoreRegister()) {
1913 StoreToOffset(kStoreWord, reg.AsCoreRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001914 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001915 } else if (reg.IsFRegister()) {
1916 StoreSToOffset(reg.AsFRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001917 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001918 } else if (reg.IsDRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001919 StoreDToOffset(reg.AsOverlappingDRegisterLow(), SP, offset);
1920 offset += kMipsDoublewordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00001921 }
jeffhao7fbee072012-08-24 17:56:54 -07001922 }
1923}
1924
1925void MipsAssembler::RemoveFrame(size_t frame_size,
1926 const std::vector<ManagedRegister>& callee_save_regs) {
1927 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001928 DCHECK(!overwriting_);
David Srbeckydd973932015-04-07 20:29:48 +01001929 cfi_.RememberState();
jeffhao7fbee072012-08-24 17:56:54 -07001930
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001931 // Pop callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07001932 int stack_offset = frame_size - (callee_save_regs.size() * kFramePointerSize) - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001933 for (size_t i = 0; i < callee_save_regs.size(); ++i) {
1934 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
1935 LoadFromOffset(kLoadWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001936 cfi_.Restore(DWARFReg(reg));
Ian Rogers790a6b72014-04-01 10:36:00 -07001937 stack_offset += kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07001938 }
1939 LoadFromOffset(kLoadWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01001940 cfi_.Restore(DWARFReg(RA));
jeffhao7fbee072012-08-24 17:56:54 -07001941
1942 // Decrease frame to required size.
1943 DecreaseFrameSize(frame_size);
jeffhao07030602012-09-26 14:33:14 -07001944
1945 // Then jump to the return address.
1946 Jr(RA);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001947 Nop();
David Srbeckydd973932015-04-07 20:29:48 +01001948
1949 // The CFI should be restored for any code that follows the exit block.
1950 cfi_.RestoreState();
1951 cfi_.DefCFAOffset(frame_size);
jeffhao7fbee072012-08-24 17:56:54 -07001952}
1953
1954void MipsAssembler::IncreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001955 CHECK_ALIGNED(adjust, kFramePointerSize);
1956 Addiu32(SP, SP, -adjust);
David Srbeckydd973932015-04-07 20:29:48 +01001957 cfi_.AdjustCFAOffset(adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001958 if (overwriting_) {
1959 cfi_.OverrideDelayedPC(overwrite_location_);
1960 }
jeffhao7fbee072012-08-24 17:56:54 -07001961}
1962
1963void MipsAssembler::DecreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001964 CHECK_ALIGNED(adjust, kFramePointerSize);
1965 Addiu32(SP, SP, adjust);
David Srbeckydd973932015-04-07 20:29:48 +01001966 cfi_.AdjustCFAOffset(-adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01001967 if (overwriting_) {
1968 cfi_.OverrideDelayedPC(overwrite_location_);
1969 }
jeffhao7fbee072012-08-24 17:56:54 -07001970}
1971
1972void MipsAssembler::Store(FrameOffset dest, ManagedRegister msrc, size_t size) {
1973 MipsManagedRegister src = msrc.AsMips();
1974 if (src.IsNoRegister()) {
1975 CHECK_EQ(0u, size);
1976 } else if (src.IsCoreRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001977 CHECK_EQ(kMipsWordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07001978 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
1979 } else if (src.IsRegisterPair()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001980 CHECK_EQ(kMipsDoublewordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07001981 StoreToOffset(kStoreWord, src.AsRegisterPairLow(), SP, dest.Int32Value());
1982 StoreToOffset(kStoreWord, src.AsRegisterPairHigh(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001983 SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001984 } else if (src.IsFRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001985 if (size == kMipsWordSize) {
1986 StoreSToOffset(src.AsFRegister(), SP, dest.Int32Value());
1987 } else {
1988 CHECK_EQ(kMipsDoublewordSize, size);
1989 StoreDToOffset(src.AsFRegister(), SP, dest.Int32Value());
1990 }
jeffhao7fbee072012-08-24 17:56:54 -07001991 }
1992}
1993
1994void MipsAssembler::StoreRef(FrameOffset dest, ManagedRegister msrc) {
1995 MipsManagedRegister src = msrc.AsMips();
1996 CHECK(src.IsCoreRegister());
1997 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
1998}
1999
2000void MipsAssembler::StoreRawPtr(FrameOffset dest, ManagedRegister msrc) {
2001 MipsManagedRegister src = msrc.AsMips();
2002 CHECK(src.IsCoreRegister());
2003 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2004}
2005
2006void MipsAssembler::StoreImmediateToFrame(FrameOffset dest, uint32_t imm,
2007 ManagedRegister mscratch) {
2008 MipsManagedRegister scratch = mscratch.AsMips();
2009 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002010 LoadConst32(scratch.AsCoreRegister(), imm);
jeffhao7fbee072012-08-24 17:56:54 -07002011 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2012}
2013
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002014void MipsAssembler::StoreImmediateToThread32(ThreadOffset<kMipsWordSize> dest, uint32_t imm,
jeffhao7fbee072012-08-24 17:56:54 -07002015 ManagedRegister mscratch) {
2016 MipsManagedRegister scratch = mscratch.AsMips();
2017 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002018 // Is this function even referenced anywhere else in the code?
2019 LoadConst32(scratch.AsCoreRegister(), imm);
2020 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), S1, dest.Int32Value());
2021}
2022
2023void MipsAssembler::StoreStackOffsetToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2024 FrameOffset fr_offs,
2025 ManagedRegister mscratch) {
2026 MipsManagedRegister scratch = mscratch.AsMips();
2027 CHECK(scratch.IsCoreRegister()) << scratch;
2028 Addiu32(scratch.AsCoreRegister(), SP, fr_offs.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002029 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2030 S1, thr_offs.Int32Value());
2031}
2032
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002033void MipsAssembler::StoreStackPointerToThread32(ThreadOffset<kMipsWordSize> thr_offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002034 StoreToOffset(kStoreWord, SP, S1, thr_offs.Int32Value());
2035}
2036
2037void MipsAssembler::StoreSpanning(FrameOffset dest, ManagedRegister msrc,
2038 FrameOffset in_off, ManagedRegister mscratch) {
2039 MipsManagedRegister src = msrc.AsMips();
2040 MipsManagedRegister scratch = mscratch.AsMips();
2041 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2042 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, in_off.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002043 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002044}
2045
2046void MipsAssembler::Load(ManagedRegister mdest, FrameOffset src, size_t size) {
2047 return EmitLoad(mdest, SP, src.Int32Value(), size);
2048}
2049
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002050void MipsAssembler::LoadFromThread32(ManagedRegister mdest,
2051 ThreadOffset<kMipsWordSize> src, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002052 return EmitLoad(mdest, S1, src.Int32Value(), size);
2053}
2054
2055void MipsAssembler::LoadRef(ManagedRegister mdest, FrameOffset src) {
2056 MipsManagedRegister dest = mdest.AsMips();
2057 CHECK(dest.IsCoreRegister());
2058 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), SP, src.Int32Value());
2059}
2060
Mathieu Chartiere401d142015-04-22 13:56:20 -07002061void MipsAssembler::LoadRef(ManagedRegister mdest, ManagedRegister base, MemberOffset offs,
Roland Levillain4d027112015-07-01 15:41:14 +01002062 bool unpoison_reference) {
jeffhao7fbee072012-08-24 17:56:54 -07002063 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002064 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002065 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2066 base.AsMips().AsCoreRegister(), offs.Int32Value());
Roland Levillain4d027112015-07-01 15:41:14 +01002067 if (kPoisonHeapReferences && unpoison_reference) {
Hiroshi Yamauchie63a7452014-02-27 14:44:36 -08002068 Subu(dest.AsCoreRegister(), ZERO, dest.AsCoreRegister());
2069 }
jeffhao7fbee072012-08-24 17:56:54 -07002070}
2071
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002072void MipsAssembler::LoadRawPtr(ManagedRegister mdest, ManagedRegister base, Offset offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002073 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002074 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002075 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2076 base.AsMips().AsCoreRegister(), offs.Int32Value());
2077}
2078
Ian Rogersdd7624d2014-03-14 17:43:00 -07002079void MipsAssembler::LoadRawPtrFromThread32(ManagedRegister mdest,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002080 ThreadOffset<kMipsWordSize> offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002081 MipsManagedRegister dest = mdest.AsMips();
2082 CHECK(dest.IsCoreRegister());
2083 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), S1, offs.Int32Value());
2084}
2085
2086void MipsAssembler::SignExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2087 UNIMPLEMENTED(FATAL) << "no sign extension necessary for mips";
2088}
2089
2090void MipsAssembler::ZeroExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2091 UNIMPLEMENTED(FATAL) << "no zero extension necessary for mips";
2092}
2093
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002094void MipsAssembler::Move(ManagedRegister mdest, ManagedRegister msrc, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002095 MipsManagedRegister dest = mdest.AsMips();
2096 MipsManagedRegister src = msrc.AsMips();
2097 if (!dest.Equals(src)) {
2098 if (dest.IsCoreRegister()) {
2099 CHECK(src.IsCoreRegister()) << src;
2100 Move(dest.AsCoreRegister(), src.AsCoreRegister());
2101 } else if (dest.IsFRegister()) {
2102 CHECK(src.IsFRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002103 if (size == kMipsWordSize) {
2104 MovS(dest.AsFRegister(), src.AsFRegister());
2105 } else {
2106 CHECK_EQ(kMipsDoublewordSize, size);
2107 MovD(dest.AsFRegister(), src.AsFRegister());
2108 }
jeffhao7fbee072012-08-24 17:56:54 -07002109 } else if (dest.IsDRegister()) {
2110 CHECK(src.IsDRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002111 MovD(dest.AsOverlappingDRegisterLow(), src.AsOverlappingDRegisterLow());
jeffhao7fbee072012-08-24 17:56:54 -07002112 } else {
2113 CHECK(dest.IsRegisterPair()) << dest;
2114 CHECK(src.IsRegisterPair()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002115 // Ensure that the first move doesn't clobber the input of the second.
jeffhao7fbee072012-08-24 17:56:54 -07002116 if (src.AsRegisterPairHigh() != dest.AsRegisterPairLow()) {
2117 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2118 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2119 } else {
2120 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2121 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2122 }
2123 }
2124 }
2125}
2126
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002127void MipsAssembler::CopyRef(FrameOffset dest, FrameOffset src, ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002128 MipsManagedRegister scratch = mscratch.AsMips();
2129 CHECK(scratch.IsCoreRegister()) << scratch;
2130 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2131 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2132}
2133
Ian Rogersdd7624d2014-03-14 17:43:00 -07002134void MipsAssembler::CopyRawPtrFromThread32(FrameOffset fr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002135 ThreadOffset<kMipsWordSize> thr_offs,
2136 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002137 MipsManagedRegister scratch = mscratch.AsMips();
2138 CHECK(scratch.IsCoreRegister()) << scratch;
2139 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2140 S1, thr_offs.Int32Value());
2141 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2142 SP, fr_offs.Int32Value());
2143}
2144
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002145void MipsAssembler::CopyRawPtrToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2146 FrameOffset fr_offs,
2147 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002148 MipsManagedRegister scratch = mscratch.AsMips();
2149 CHECK(scratch.IsCoreRegister()) << scratch;
2150 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2151 SP, fr_offs.Int32Value());
2152 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2153 S1, thr_offs.Int32Value());
2154}
2155
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002156void MipsAssembler::Copy(FrameOffset dest, FrameOffset src, ManagedRegister mscratch, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002157 MipsManagedRegister scratch = mscratch.AsMips();
2158 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002159 CHECK(size == kMipsWordSize || size == kMipsDoublewordSize) << size;
2160 if (size == kMipsWordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002161 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2162 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002163 } else if (size == kMipsDoublewordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002164 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2165 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002166 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value() + kMipsWordSize);
2167 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002168 }
2169}
2170
2171void MipsAssembler::Copy(FrameOffset dest, ManagedRegister src_base, Offset src_offset,
2172 ManagedRegister mscratch, size_t size) {
2173 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002174 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002175 LoadFromOffset(kLoadWord, scratch, src_base.AsMips().AsCoreRegister(), src_offset.Int32Value());
2176 StoreToOffset(kStoreWord, scratch, SP, dest.Int32Value());
2177}
2178
2179void MipsAssembler::Copy(ManagedRegister dest_base, Offset dest_offset, FrameOffset src,
2180 ManagedRegister mscratch, size_t size) {
2181 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002182 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002183 LoadFromOffset(kLoadWord, scratch, SP, src.Int32Value());
2184 StoreToOffset(kStoreWord, scratch, dest_base.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2185}
2186
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002187void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2188 FrameOffset src_base ATTRIBUTE_UNUSED,
2189 Offset src_offset ATTRIBUTE_UNUSED,
2190 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2191 size_t size ATTRIBUTE_UNUSED) {
2192 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002193}
2194
2195void MipsAssembler::Copy(ManagedRegister dest, Offset dest_offset,
2196 ManagedRegister src, Offset src_offset,
2197 ManagedRegister mscratch, size_t size) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002198 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002199 Register scratch = mscratch.AsMips().AsCoreRegister();
2200 LoadFromOffset(kLoadWord, scratch, src.AsMips().AsCoreRegister(), src_offset.Int32Value());
2201 StoreToOffset(kStoreWord, scratch, dest.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2202}
2203
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002204void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2205 Offset dest_offset ATTRIBUTE_UNUSED,
2206 FrameOffset src ATTRIBUTE_UNUSED,
2207 Offset src_offset ATTRIBUTE_UNUSED,
2208 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2209 size_t size ATTRIBUTE_UNUSED) {
2210 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002211}
2212
2213void MipsAssembler::MemoryBarrier(ManagedRegister) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002214 // TODO: sync?
2215 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002216}
2217
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002218void MipsAssembler::CreateHandleScopeEntry(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002219 FrameOffset handle_scope_offset,
2220 ManagedRegister min_reg,
2221 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002222 MipsManagedRegister out_reg = mout_reg.AsMips();
2223 MipsManagedRegister in_reg = min_reg.AsMips();
2224 CHECK(in_reg.IsNoRegister() || in_reg.IsCoreRegister()) << in_reg;
2225 CHECK(out_reg.IsCoreRegister()) << out_reg;
2226 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002227 MipsLabel null_arg;
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002228 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2229 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002230 // E.g. out_reg = (handle == 0) ? 0 : (SP+handle_offset).
jeffhao7fbee072012-08-24 17:56:54 -07002231 if (in_reg.IsNoRegister()) {
2232 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002233 SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002234 in_reg = out_reg;
2235 }
2236 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002237 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002238 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002239 Beqz(in_reg.AsCoreRegister(), &null_arg);
2240 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2241 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002242 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002243 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002244 }
2245}
2246
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002247void MipsAssembler::CreateHandleScopeEntry(FrameOffset out_off,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002248 FrameOffset handle_scope_offset,
2249 ManagedRegister mscratch,
2250 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002251 MipsManagedRegister scratch = mscratch.AsMips();
2252 CHECK(scratch.IsCoreRegister()) << scratch;
2253 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002254 MipsLabel null_arg;
2255 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002256 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2257 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002258 // E.g. scratch = (scratch == 0) ? 0 : (SP+handle_scope_offset).
2259 Beqz(scratch.AsCoreRegister(), &null_arg);
2260 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2261 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002262 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002263 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002264 }
2265 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, out_off.Int32Value());
2266}
2267
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002268// Given a handle scope entry, load the associated reference.
2269void MipsAssembler::LoadReferenceFromHandleScope(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002270 ManagedRegister min_reg) {
jeffhao7fbee072012-08-24 17:56:54 -07002271 MipsManagedRegister out_reg = mout_reg.AsMips();
2272 MipsManagedRegister in_reg = min_reg.AsMips();
2273 CHECK(out_reg.IsCoreRegister()) << out_reg;
2274 CHECK(in_reg.IsCoreRegister()) << in_reg;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002275 MipsLabel null_arg;
jeffhao7fbee072012-08-24 17:56:54 -07002276 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002277 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002278 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002279 Beqz(in_reg.AsCoreRegister(), &null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002280 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
2281 in_reg.AsCoreRegister(), 0);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002282 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002283}
2284
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002285void MipsAssembler::VerifyObject(ManagedRegister src ATTRIBUTE_UNUSED,
2286 bool could_be_null ATTRIBUTE_UNUSED) {
2287 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002288}
2289
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002290void MipsAssembler::VerifyObject(FrameOffset src ATTRIBUTE_UNUSED,
2291 bool could_be_null ATTRIBUTE_UNUSED) {
2292 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002293}
2294
2295void MipsAssembler::Call(ManagedRegister mbase, Offset offset, ManagedRegister mscratch) {
2296 MipsManagedRegister base = mbase.AsMips();
2297 MipsManagedRegister scratch = mscratch.AsMips();
2298 CHECK(base.IsCoreRegister()) << base;
2299 CHECK(scratch.IsCoreRegister()) << scratch;
2300 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2301 base.AsCoreRegister(), offset.Int32Value());
2302 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002303 Nop();
2304 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002305}
2306
2307void MipsAssembler::Call(FrameOffset base, Offset offset, ManagedRegister mscratch) {
2308 MipsManagedRegister scratch = mscratch.AsMips();
2309 CHECK(scratch.IsCoreRegister()) << scratch;
2310 // Call *(*(SP + base) + offset)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002311 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, base.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002312 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2313 scratch.AsCoreRegister(), offset.Int32Value());
2314 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002315 Nop();
2316 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002317}
2318
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002319void MipsAssembler::CallFromThread32(ThreadOffset<kMipsWordSize> offset ATTRIBUTE_UNUSED,
2320 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
Ian Rogers468532e2013-08-05 10:56:33 -07002321 UNIMPLEMENTED(FATAL) << "no mips implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002322}
2323
2324void MipsAssembler::GetCurrentThread(ManagedRegister tr) {
2325 Move(tr.AsMips().AsCoreRegister(), S1);
2326}
2327
2328void MipsAssembler::GetCurrentThread(FrameOffset offset,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002329 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
jeffhao7fbee072012-08-24 17:56:54 -07002330 StoreToOffset(kStoreWord, S1, SP, offset.Int32Value());
2331}
2332
jeffhao7fbee072012-08-24 17:56:54 -07002333void MipsAssembler::ExceptionPoll(ManagedRegister mscratch, size_t stack_adjust) {
2334 MipsManagedRegister scratch = mscratch.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002335 exception_blocks_.emplace_back(scratch, stack_adjust);
jeffhao7fbee072012-08-24 17:56:54 -07002336 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002337 S1, Thread::ExceptionOffset<kMipsWordSize>().Int32Value());
2338 // TODO: on MIPS32R6 prefer Bnezc(scratch.AsCoreRegister(), slow.Entry());
2339 // as the NAL instruction (occurring in long R2 branches) may become deprecated.
2340 // For now use common for R2 and R6 instructions as this code must execute on both.
2341 Bnez(scratch.AsCoreRegister(), exception_blocks_.back().Entry());
jeffhao7fbee072012-08-24 17:56:54 -07002342}
2343
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002344void MipsAssembler::EmitExceptionPoll(MipsExceptionSlowPath* exception) {
2345 Bind(exception->Entry());
2346 if (exception->stack_adjust_ != 0) { // Fix up the frame.
2347 DecreaseFrameSize(exception->stack_adjust_);
jeffhao7fbee072012-08-24 17:56:54 -07002348 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002349 // Pass exception object as argument.
2350 // Don't care about preserving A0 as this call won't return.
2351 CheckEntrypointTypes<kQuickDeliverException, void, mirror::Object*>();
2352 Move(A0, exception->scratch_.AsCoreRegister());
2353 // Set up call to Thread::Current()->pDeliverException.
2354 LoadFromOffset(kLoadWord, T9, S1,
2355 QUICK_ENTRYPOINT_OFFSET(kMipsWordSize, pDeliverException).Int32Value());
2356 Jr(T9);
2357 Nop();
2358
2359 // Call never returns.
2360 Break();
jeffhao7fbee072012-08-24 17:56:54 -07002361}
2362
2363} // namespace mips
2364} // namespace art