blob: e6b32def55d2e9e7d8f9313a4b23b7a4290c3ef7 [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
Andreas Gampe542451c2016-07-26 09:02:02 -070029static_assert(static_cast<size_t>(kMipsPointerSize) == kMipsWordSize,
30 "Unexpected Mips pointer size.");
31static_assert(kMipsPointerSize == PointerSize::k32, "Unexpected Mips pointer size.");
32
33
jeffhao7fbee072012-08-24 17:56:54 -070034std::ostream& operator<<(std::ostream& os, const DRegister& rhs) {
35 if (rhs >= D0 && rhs < kNumberOfDRegisters) {
36 os << "d" << static_cast<int>(rhs);
37 } else {
38 os << "DRegister[" << static_cast<int>(rhs) << "]";
39 }
40 return os;
41}
42
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020043void MipsAssembler::FinalizeCode() {
44 for (auto& exception_block : exception_blocks_) {
45 EmitExceptionPoll(&exception_block);
46 }
Alexey Frunzee3fb2452016-05-10 16:08:05 -070047 EmitLiterals();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020048 PromoteBranches();
49}
50
51void MipsAssembler::FinalizeInstructions(const MemoryRegion& region) {
Vladimir Marko10ef6942015-10-22 15:25:54 +010052 size_t number_of_delayed_adjust_pcs = cfi().NumberOfDelayedAdvancePCs();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020053 EmitBranches();
54 Assembler::FinalizeInstructions(region);
Vladimir Marko10ef6942015-10-22 15:25:54 +010055 PatchCFI(number_of_delayed_adjust_pcs);
56}
57
58void MipsAssembler::PatchCFI(size_t number_of_delayed_adjust_pcs) {
59 if (cfi().NumberOfDelayedAdvancePCs() == 0u) {
60 DCHECK_EQ(number_of_delayed_adjust_pcs, 0u);
61 return;
62 }
63
64 typedef DebugFrameOpCodeWriterForAssembler::DelayedAdvancePC DelayedAdvancePC;
65 const auto data = cfi().ReleaseStreamAndPrepareForDelayedAdvancePC();
66 const std::vector<uint8_t>& old_stream = data.first;
67 const std::vector<DelayedAdvancePC>& advances = data.second;
68
69 // PCs recorded before EmitBranches() need to be adjusted.
70 // PCs recorded during EmitBranches() are already adjusted.
71 // Both ranges are separately sorted but they may overlap.
72 if (kIsDebugBuild) {
73 auto cmp = [](const DelayedAdvancePC& lhs, const DelayedAdvancePC& rhs) {
74 return lhs.pc < rhs.pc;
75 };
76 CHECK(std::is_sorted(advances.begin(), advances.begin() + number_of_delayed_adjust_pcs, cmp));
77 CHECK(std::is_sorted(advances.begin() + number_of_delayed_adjust_pcs, advances.end(), cmp));
78 }
79
80 // Append initial CFI data if any.
81 size_t size = advances.size();
82 DCHECK_NE(size, 0u);
83 cfi().AppendRawData(old_stream, 0u, advances[0].stream_pos);
84 // Emit PC adjustments interleaved with the old CFI stream.
85 size_t adjust_pos = 0u;
86 size_t late_emit_pos = number_of_delayed_adjust_pcs;
87 while (adjust_pos != number_of_delayed_adjust_pcs || late_emit_pos != size) {
88 size_t adjusted_pc = (adjust_pos != number_of_delayed_adjust_pcs)
89 ? GetAdjustedPosition(advances[adjust_pos].pc)
90 : static_cast<size_t>(-1);
91 size_t late_emit_pc = (late_emit_pos != size)
92 ? advances[late_emit_pos].pc
93 : static_cast<size_t>(-1);
94 size_t advance_pc = std::min(adjusted_pc, late_emit_pc);
95 DCHECK_NE(advance_pc, static_cast<size_t>(-1));
96 size_t entry = (adjusted_pc <= late_emit_pc) ? adjust_pos : late_emit_pos;
97 if (adjusted_pc <= late_emit_pc) {
98 ++adjust_pos;
99 } else {
100 ++late_emit_pos;
101 }
102 cfi().AdvancePC(advance_pc);
103 size_t end_pos = (entry + 1u == size) ? old_stream.size() : advances[entry + 1u].stream_pos;
104 cfi().AppendRawData(old_stream, advances[entry].stream_pos, end_pos);
105 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200106}
107
108void MipsAssembler::EmitBranches() {
109 CHECK(!overwriting_);
110 // Switch from appending instructions at the end of the buffer to overwriting
111 // existing instructions (branch placeholders) in the buffer.
112 overwriting_ = true;
113 for (auto& branch : branches_) {
114 EmitBranch(&branch);
115 }
116 overwriting_ = false;
117}
118
119void MipsAssembler::Emit(uint32_t value) {
120 if (overwriting_) {
121 // Branches to labels are emitted into their placeholders here.
122 buffer_.Store<uint32_t>(overwrite_location_, value);
123 overwrite_location_ += sizeof(uint32_t);
124 } else {
125 // Other instructions are simply appended at the end here.
126 AssemblerBuffer::EnsureCapacity ensured(&buffer_);
127 buffer_.Emit<uint32_t>(value);
128 }
jeffhao7fbee072012-08-24 17:56:54 -0700129}
130
131void MipsAssembler::EmitR(int opcode, Register rs, Register rt, Register rd, int shamt, int funct) {
132 CHECK_NE(rs, kNoRegister);
133 CHECK_NE(rt, kNoRegister);
134 CHECK_NE(rd, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200135 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
136 static_cast<uint32_t>(rs) << kRsShift |
137 static_cast<uint32_t>(rt) << kRtShift |
138 static_cast<uint32_t>(rd) << kRdShift |
139 shamt << kShamtShift |
140 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700141 Emit(encoding);
142}
143
144void MipsAssembler::EmitI(int opcode, Register rs, Register rt, uint16_t imm) {
145 CHECK_NE(rs, kNoRegister);
146 CHECK_NE(rt, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200147 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
148 static_cast<uint32_t>(rs) << kRsShift |
149 static_cast<uint32_t>(rt) << kRtShift |
150 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700151 Emit(encoding);
152}
153
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200154void MipsAssembler::EmitI21(int opcode, Register rs, uint32_t imm21) {
155 CHECK_NE(rs, kNoRegister);
156 CHECK(IsUint<21>(imm21)) << imm21;
157 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
158 static_cast<uint32_t>(rs) << kRsShift |
159 imm21;
jeffhao7fbee072012-08-24 17:56:54 -0700160 Emit(encoding);
161}
162
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200163void MipsAssembler::EmitI26(int opcode, uint32_t imm26) {
164 CHECK(IsUint<26>(imm26)) << imm26;
165 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift | imm26;
166 Emit(encoding);
167}
168
169void MipsAssembler::EmitFR(int opcode, int fmt, FRegister ft, FRegister fs, FRegister fd,
170 int funct) {
jeffhao7fbee072012-08-24 17:56:54 -0700171 CHECK_NE(ft, kNoFRegister);
172 CHECK_NE(fs, kNoFRegister);
173 CHECK_NE(fd, kNoFRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200174 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
175 fmt << kFmtShift |
176 static_cast<uint32_t>(ft) << kFtShift |
177 static_cast<uint32_t>(fs) << kFsShift |
178 static_cast<uint32_t>(fd) << kFdShift |
179 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700180 Emit(encoding);
181}
182
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200183void MipsAssembler::EmitFI(int opcode, int fmt, FRegister ft, uint16_t imm) {
184 CHECK_NE(ft, kNoFRegister);
185 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
186 fmt << kFmtShift |
187 static_cast<uint32_t>(ft) << kFtShift |
188 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700189 Emit(encoding);
190}
191
jeffhao7fbee072012-08-24 17:56:54 -0700192void MipsAssembler::Addu(Register rd, Register rs, Register rt) {
193 EmitR(0, rs, rt, rd, 0, 0x21);
194}
195
jeffhao7fbee072012-08-24 17:56:54 -0700196void MipsAssembler::Addiu(Register rt, Register rs, uint16_t imm16) {
197 EmitI(0x9, rs, rt, imm16);
198}
199
jeffhao7fbee072012-08-24 17:56:54 -0700200void MipsAssembler::Subu(Register rd, Register rs, Register rt) {
201 EmitR(0, rs, rt, rd, 0, 0x23);
202}
203
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200204void MipsAssembler::MultR2(Register rs, Register rt) {
205 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700206 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x18);
207}
208
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200209void MipsAssembler::MultuR2(Register rs, Register rt) {
210 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700211 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x19);
212}
213
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200214void MipsAssembler::DivR2(Register rs, Register rt) {
215 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700216 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1a);
217}
218
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200219void MipsAssembler::DivuR2(Register rs, Register rt) {
220 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700221 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1b);
222}
223
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200224void MipsAssembler::MulR2(Register rd, Register rs, Register rt) {
225 CHECK(!IsR6());
226 EmitR(0x1c, rs, rt, rd, 0, 2);
227}
228
229void MipsAssembler::DivR2(Register rd, Register rs, Register rt) {
230 CHECK(!IsR6());
231 DivR2(rs, rt);
232 Mflo(rd);
233}
234
235void MipsAssembler::ModR2(Register rd, Register rs, Register rt) {
236 CHECK(!IsR6());
237 DivR2(rs, rt);
238 Mfhi(rd);
239}
240
241void MipsAssembler::DivuR2(Register rd, Register rs, Register rt) {
242 CHECK(!IsR6());
243 DivuR2(rs, rt);
244 Mflo(rd);
245}
246
247void MipsAssembler::ModuR2(Register rd, Register rs, Register rt) {
248 CHECK(!IsR6());
249 DivuR2(rs, rt);
250 Mfhi(rd);
251}
252
253void MipsAssembler::MulR6(Register rd, Register rs, Register rt) {
254 CHECK(IsR6());
255 EmitR(0, rs, rt, rd, 2, 0x18);
256}
257
Alexey Frunze7e99e052015-11-24 19:28:01 -0800258void MipsAssembler::MuhR6(Register rd, Register rs, Register rt) {
259 CHECK(IsR6());
260 EmitR(0, rs, rt, rd, 3, 0x18);
261}
262
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200263void MipsAssembler::MuhuR6(Register rd, Register rs, Register rt) {
264 CHECK(IsR6());
265 EmitR(0, rs, rt, rd, 3, 0x19);
266}
267
268void MipsAssembler::DivR6(Register rd, Register rs, Register rt) {
269 CHECK(IsR6());
270 EmitR(0, rs, rt, rd, 2, 0x1a);
271}
272
273void MipsAssembler::ModR6(Register rd, Register rs, Register rt) {
274 CHECK(IsR6());
275 EmitR(0, rs, rt, rd, 3, 0x1a);
276}
277
278void MipsAssembler::DivuR6(Register rd, Register rs, Register rt) {
279 CHECK(IsR6());
280 EmitR(0, rs, rt, rd, 2, 0x1b);
281}
282
283void MipsAssembler::ModuR6(Register rd, Register rs, Register rt) {
284 CHECK(IsR6());
285 EmitR(0, rs, rt, rd, 3, 0x1b);
286}
287
jeffhao7fbee072012-08-24 17:56:54 -0700288void MipsAssembler::And(Register rd, Register rs, Register rt) {
289 EmitR(0, rs, rt, rd, 0, 0x24);
290}
291
292void MipsAssembler::Andi(Register rt, Register rs, uint16_t imm16) {
293 EmitI(0xc, rs, rt, imm16);
294}
295
296void MipsAssembler::Or(Register rd, Register rs, Register rt) {
297 EmitR(0, rs, rt, rd, 0, 0x25);
298}
299
300void MipsAssembler::Ori(Register rt, Register rs, uint16_t imm16) {
301 EmitI(0xd, rs, rt, imm16);
302}
303
304void MipsAssembler::Xor(Register rd, Register rs, Register rt) {
305 EmitR(0, rs, rt, rd, 0, 0x26);
306}
307
308void MipsAssembler::Xori(Register rt, Register rs, uint16_t imm16) {
309 EmitI(0xe, rs, rt, imm16);
310}
311
312void MipsAssembler::Nor(Register rd, Register rs, Register rt) {
313 EmitR(0, rs, rt, rd, 0, 0x27);
314}
315
Chris Larsene3845472015-11-18 12:27:15 -0800316void MipsAssembler::Movz(Register rd, Register rs, Register rt) {
317 CHECK(!IsR6());
318 EmitR(0, rs, rt, rd, 0, 0x0A);
319}
320
321void MipsAssembler::Movn(Register rd, Register rs, Register rt) {
322 CHECK(!IsR6());
323 EmitR(0, rs, rt, rd, 0, 0x0B);
324}
325
326void MipsAssembler::Seleqz(Register rd, Register rs, Register rt) {
327 CHECK(IsR6());
328 EmitR(0, rs, rt, rd, 0, 0x35);
329}
330
331void MipsAssembler::Selnez(Register rd, Register rs, Register rt) {
332 CHECK(IsR6());
333 EmitR(0, rs, rt, rd, 0, 0x37);
334}
335
336void MipsAssembler::ClzR6(Register rd, Register rs) {
337 CHECK(IsR6());
338 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x10);
339}
340
341void MipsAssembler::ClzR2(Register rd, Register rs) {
342 CHECK(!IsR6());
343 EmitR(0x1C, rs, rd, rd, 0, 0x20);
344}
345
346void MipsAssembler::CloR6(Register rd, Register rs) {
347 CHECK(IsR6());
348 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x11);
349}
350
351void MipsAssembler::CloR2(Register rd, Register rs) {
352 CHECK(!IsR6());
353 EmitR(0x1C, rs, rd, rd, 0, 0x21);
354}
355
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200356void MipsAssembler::Seb(Register rd, Register rt) {
357 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x10, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700358}
359
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200360void MipsAssembler::Seh(Register rd, Register rt) {
361 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x18, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700362}
363
Chris Larsen3f8bf652015-10-28 10:08:56 -0700364void MipsAssembler::Wsbh(Register rd, Register rt) {
365 EmitR(0x1f, static_cast<Register>(0), rt, rd, 2, 0x20);
366}
367
Chris Larsen70014c82015-11-18 12:26:08 -0800368void MipsAssembler::Bitswap(Register rd, Register rt) {
369 CHECK(IsR6());
370 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x0, 0x20);
371}
372
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200373void MipsAssembler::Sll(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700374 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200375 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x00);
jeffhao7fbee072012-08-24 17:56:54 -0700376}
377
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200378void MipsAssembler::Srl(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700379 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200380 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x02);
381}
382
Chris Larsen3f8bf652015-10-28 10:08:56 -0700383void MipsAssembler::Rotr(Register rd, Register rt, int shamt) {
384 CHECK(IsUint<5>(shamt)) << shamt;
385 EmitR(0, static_cast<Register>(1), rt, rd, shamt, 0x02);
386}
387
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200388void MipsAssembler::Sra(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700389 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200390 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x03);
391}
392
393void MipsAssembler::Sllv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700394 EmitR(0, rs, rt, rd, 0, 0x04);
395}
396
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200397void MipsAssembler::Srlv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700398 EmitR(0, rs, rt, rd, 0, 0x06);
399}
400
Chris Larsene16ce5a2015-11-18 12:30:20 -0800401void MipsAssembler::Rotrv(Register rd, Register rt, Register rs) {
402 EmitR(0, rs, rt, rd, 1, 0x06);
403}
404
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200405void MipsAssembler::Srav(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700406 EmitR(0, rs, rt, rd, 0, 0x07);
407}
408
Alexey Frunze5c7aed32015-11-25 19:41:54 -0800409void MipsAssembler::Ext(Register rd, Register rt, int pos, int size) {
410 CHECK(IsUint<5>(pos)) << pos;
411 CHECK(0 < size && size <= 32) << size;
412 CHECK(0 < pos + size && pos + size <= 32) << pos << " + " << size;
413 EmitR(0x1f, rt, rd, static_cast<Register>(size - 1), pos, 0x00);
414}
415
416void MipsAssembler::Ins(Register rd, Register rt, int pos, int size) {
417 CHECK(IsUint<5>(pos)) << pos;
418 CHECK(0 < size && size <= 32) << size;
419 CHECK(0 < pos + size && pos + size <= 32) << pos << " + " << size;
420 EmitR(0x1f, rt, rd, static_cast<Register>(pos + size - 1), pos, 0x04);
421}
422
jeffhao7fbee072012-08-24 17:56:54 -0700423void MipsAssembler::Lb(Register rt, Register rs, uint16_t imm16) {
424 EmitI(0x20, rs, rt, imm16);
425}
426
427void MipsAssembler::Lh(Register rt, Register rs, uint16_t imm16) {
428 EmitI(0x21, rs, rt, imm16);
429}
430
431void MipsAssembler::Lw(Register rt, Register rs, uint16_t imm16) {
432 EmitI(0x23, rs, rt, imm16);
433}
434
Chris Larsen3acee732015-11-18 13:31:08 -0800435void MipsAssembler::Lwl(Register rt, Register rs, uint16_t imm16) {
436 CHECK(!IsR6());
437 EmitI(0x22, rs, rt, imm16);
438}
439
440void MipsAssembler::Lwr(Register rt, Register rs, uint16_t imm16) {
441 CHECK(!IsR6());
442 EmitI(0x26, rs, rt, imm16);
443}
444
jeffhao7fbee072012-08-24 17:56:54 -0700445void MipsAssembler::Lbu(Register rt, Register rs, uint16_t imm16) {
446 EmitI(0x24, rs, rt, imm16);
447}
448
449void MipsAssembler::Lhu(Register rt, Register rs, uint16_t imm16) {
450 EmitI(0x25, rs, rt, imm16);
451}
452
Alexey Frunzee3fb2452016-05-10 16:08:05 -0700453void MipsAssembler::Lwpc(Register rs, uint32_t imm19) {
454 CHECK(IsR6());
455 CHECK(IsUint<19>(imm19)) << imm19;
456 EmitI21(0x3B, rs, (0x01 << 19) | imm19);
457}
458
jeffhao7fbee072012-08-24 17:56:54 -0700459void MipsAssembler::Lui(Register rt, uint16_t imm16) {
460 EmitI(0xf, static_cast<Register>(0), rt, imm16);
461}
462
Alexey Frunzecad3a4c2016-06-07 23:40:37 -0700463void MipsAssembler::Aui(Register rt, Register rs, uint16_t imm16) {
464 CHECK(IsR6());
465 EmitI(0xf, rs, rt, imm16);
466}
467
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200468void MipsAssembler::Sync(uint32_t stype) {
469 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0),
470 stype & 0x1f, 0xf);
471}
472
jeffhao7fbee072012-08-24 17:56:54 -0700473void MipsAssembler::Mfhi(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200474 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700475 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x10);
476}
477
478void MipsAssembler::Mflo(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200479 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700480 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x12);
481}
482
483void MipsAssembler::Sb(Register rt, Register rs, uint16_t imm16) {
484 EmitI(0x28, rs, rt, imm16);
485}
486
487void MipsAssembler::Sh(Register rt, Register rs, uint16_t imm16) {
488 EmitI(0x29, rs, rt, imm16);
489}
490
491void MipsAssembler::Sw(Register rt, Register rs, uint16_t imm16) {
492 EmitI(0x2b, rs, rt, imm16);
493}
494
Chris Larsen3acee732015-11-18 13:31:08 -0800495void MipsAssembler::Swl(Register rt, Register rs, uint16_t imm16) {
496 CHECK(!IsR6());
497 EmitI(0x2a, rs, rt, imm16);
498}
499
500void MipsAssembler::Swr(Register rt, Register rs, uint16_t imm16) {
501 CHECK(!IsR6());
502 EmitI(0x2e, rs, rt, imm16);
503}
504
Alexey Frunze51aff3a2016-03-17 17:21:45 -0700505void MipsAssembler::LlR2(Register rt, Register base, int16_t imm16) {
506 CHECK(!IsR6());
507 EmitI(0x30, base, rt, imm16);
508}
509
510void MipsAssembler::ScR2(Register rt, Register base, int16_t imm16) {
511 CHECK(!IsR6());
512 EmitI(0x38, base, rt, imm16);
513}
514
515void MipsAssembler::LlR6(Register rt, Register base, int16_t imm9) {
516 CHECK(IsR6());
517 CHECK(IsInt<9>(imm9));
518 EmitI(0x1f, base, rt, ((imm9 & 0x1ff) << 7) | 0x36);
519}
520
521void MipsAssembler::ScR6(Register rt, Register base, int16_t imm9) {
522 CHECK(IsR6());
523 CHECK(IsInt<9>(imm9));
524 EmitI(0x1f, base, rt, ((imm9 & 0x1ff) << 7) | 0x26);
525}
526
jeffhao7fbee072012-08-24 17:56:54 -0700527void MipsAssembler::Slt(Register rd, Register rs, Register rt) {
528 EmitR(0, rs, rt, rd, 0, 0x2a);
529}
530
531void MipsAssembler::Sltu(Register rd, Register rs, Register rt) {
532 EmitR(0, rs, rt, rd, 0, 0x2b);
533}
534
535void MipsAssembler::Slti(Register rt, Register rs, uint16_t imm16) {
536 EmitI(0xa, rs, rt, imm16);
537}
538
539void MipsAssembler::Sltiu(Register rt, Register rs, uint16_t imm16) {
540 EmitI(0xb, rs, rt, imm16);
541}
542
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200543void MipsAssembler::B(uint16_t imm16) {
544 EmitI(0x4, static_cast<Register>(0), static_cast<Register>(0), imm16);
545}
546
Alexey Frunzee3fb2452016-05-10 16:08:05 -0700547void MipsAssembler::Bal(uint16_t imm16) {
548 EmitI(0x1, static_cast<Register>(0), static_cast<Register>(0x11), imm16);
549}
550
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200551void MipsAssembler::Beq(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700552 EmitI(0x4, rs, rt, imm16);
553}
554
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200555void MipsAssembler::Bne(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700556 EmitI(0x5, rs, rt, imm16);
557}
558
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200559void MipsAssembler::Beqz(Register rt, uint16_t imm16) {
560 Beq(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700561}
562
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200563void MipsAssembler::Bnez(Register rt, uint16_t imm16) {
564 Bne(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700565}
566
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200567void MipsAssembler::Bltz(Register rt, uint16_t imm16) {
568 EmitI(0x1, rt, static_cast<Register>(0), imm16);
569}
570
571void MipsAssembler::Bgez(Register rt, uint16_t imm16) {
572 EmitI(0x1, rt, static_cast<Register>(0x1), imm16);
573}
574
575void MipsAssembler::Blez(Register rt, uint16_t imm16) {
576 EmitI(0x6, rt, static_cast<Register>(0), imm16);
577}
578
579void MipsAssembler::Bgtz(Register rt, uint16_t imm16) {
580 EmitI(0x7, rt, static_cast<Register>(0), imm16);
581}
582
Chris Larsenb74353a2015-11-20 09:07:09 -0800583void MipsAssembler::Bc1f(uint16_t imm16) {
584 Bc1f(0, imm16);
585}
586
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800587void MipsAssembler::Bc1f(int cc, uint16_t imm16) {
588 CHECK(!IsR6());
589 CHECK(IsUint<3>(cc)) << cc;
590 EmitI(0x11, static_cast<Register>(0x8), static_cast<Register>(cc << 2), imm16);
591}
592
Chris Larsenb74353a2015-11-20 09:07:09 -0800593void MipsAssembler::Bc1t(uint16_t imm16) {
594 Bc1t(0, imm16);
595}
596
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800597void MipsAssembler::Bc1t(int cc, uint16_t imm16) {
598 CHECK(!IsR6());
599 CHECK(IsUint<3>(cc)) << cc;
600 EmitI(0x11, static_cast<Register>(0x8), static_cast<Register>((cc << 2) | 1), imm16);
601}
602
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200603void MipsAssembler::J(uint32_t addr26) {
604 EmitI26(0x2, addr26);
605}
606
607void MipsAssembler::Jal(uint32_t addr26) {
608 EmitI26(0x3, addr26);
609}
610
611void MipsAssembler::Jalr(Register rd, Register rs) {
612 EmitR(0, rs, static_cast<Register>(0), rd, 0, 0x09);
jeffhao7fbee072012-08-24 17:56:54 -0700613}
614
615void MipsAssembler::Jalr(Register rs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200616 Jalr(RA, rs);
617}
618
619void MipsAssembler::Jr(Register rs) {
620 Jalr(ZERO, rs);
621}
622
623void MipsAssembler::Nal() {
624 EmitI(0x1, static_cast<Register>(0), static_cast<Register>(0x10), 0);
625}
626
627void MipsAssembler::Auipc(Register rs, uint16_t imm16) {
628 CHECK(IsR6());
629 EmitI(0x3B, rs, static_cast<Register>(0x1E), imm16);
630}
631
632void MipsAssembler::Addiupc(Register rs, uint32_t imm19) {
633 CHECK(IsR6());
634 CHECK(IsUint<19>(imm19)) << imm19;
635 EmitI21(0x3B, rs, imm19);
636}
637
638void MipsAssembler::Bc(uint32_t imm26) {
639 CHECK(IsR6());
640 EmitI26(0x32, imm26);
641}
642
Alexey Frunzee3fb2452016-05-10 16:08:05 -0700643void MipsAssembler::Balc(uint32_t imm26) {
644 CHECK(IsR6());
645 EmitI26(0x3A, imm26);
646}
647
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200648void MipsAssembler::Jic(Register rt, uint16_t imm16) {
649 CHECK(IsR6());
650 EmitI(0x36, static_cast<Register>(0), rt, imm16);
651}
652
653void MipsAssembler::Jialc(Register rt, uint16_t imm16) {
654 CHECK(IsR6());
655 EmitI(0x3E, static_cast<Register>(0), rt, imm16);
656}
657
658void MipsAssembler::Bltc(Register rs, Register rt, uint16_t imm16) {
659 CHECK(IsR6());
660 CHECK_NE(rs, ZERO);
661 CHECK_NE(rt, ZERO);
662 CHECK_NE(rs, rt);
663 EmitI(0x17, rs, rt, imm16);
664}
665
666void MipsAssembler::Bltzc(Register rt, uint16_t imm16) {
667 CHECK(IsR6());
668 CHECK_NE(rt, ZERO);
669 EmitI(0x17, rt, rt, imm16);
670}
671
672void MipsAssembler::Bgtzc(Register rt, uint16_t imm16) {
673 CHECK(IsR6());
674 CHECK_NE(rt, ZERO);
675 EmitI(0x17, static_cast<Register>(0), rt, imm16);
676}
677
678void MipsAssembler::Bgec(Register rs, Register rt, uint16_t imm16) {
679 CHECK(IsR6());
680 CHECK_NE(rs, ZERO);
681 CHECK_NE(rt, ZERO);
682 CHECK_NE(rs, rt);
683 EmitI(0x16, rs, rt, imm16);
684}
685
686void MipsAssembler::Bgezc(Register rt, uint16_t imm16) {
687 CHECK(IsR6());
688 CHECK_NE(rt, ZERO);
689 EmitI(0x16, rt, rt, imm16);
690}
691
692void MipsAssembler::Blezc(Register rt, uint16_t imm16) {
693 CHECK(IsR6());
694 CHECK_NE(rt, ZERO);
695 EmitI(0x16, static_cast<Register>(0), rt, imm16);
696}
697
698void MipsAssembler::Bltuc(Register rs, Register rt, uint16_t imm16) {
699 CHECK(IsR6());
700 CHECK_NE(rs, ZERO);
701 CHECK_NE(rt, ZERO);
702 CHECK_NE(rs, rt);
703 EmitI(0x7, rs, rt, imm16);
704}
705
706void MipsAssembler::Bgeuc(Register rs, Register rt, uint16_t imm16) {
707 CHECK(IsR6());
708 CHECK_NE(rs, ZERO);
709 CHECK_NE(rt, ZERO);
710 CHECK_NE(rs, rt);
711 EmitI(0x6, rs, rt, imm16);
712}
713
714void MipsAssembler::Beqc(Register rs, Register rt, uint16_t imm16) {
715 CHECK(IsR6());
716 CHECK_NE(rs, ZERO);
717 CHECK_NE(rt, ZERO);
718 CHECK_NE(rs, rt);
719 EmitI(0x8, std::min(rs, rt), std::max(rs, rt), imm16);
720}
721
722void MipsAssembler::Bnec(Register rs, Register rt, uint16_t imm16) {
723 CHECK(IsR6());
724 CHECK_NE(rs, ZERO);
725 CHECK_NE(rt, ZERO);
726 CHECK_NE(rs, rt);
727 EmitI(0x18, std::min(rs, rt), std::max(rs, rt), imm16);
728}
729
730void MipsAssembler::Beqzc(Register rs, uint32_t imm21) {
731 CHECK(IsR6());
732 CHECK_NE(rs, ZERO);
733 EmitI21(0x36, rs, imm21);
734}
735
736void MipsAssembler::Bnezc(Register rs, uint32_t imm21) {
737 CHECK(IsR6());
738 CHECK_NE(rs, ZERO);
739 EmitI21(0x3E, rs, imm21);
740}
741
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800742void MipsAssembler::Bc1eqz(FRegister ft, uint16_t imm16) {
743 CHECK(IsR6());
744 EmitFI(0x11, 0x9, ft, imm16);
745}
746
747void MipsAssembler::Bc1nez(FRegister ft, uint16_t imm16) {
748 CHECK(IsR6());
749 EmitFI(0x11, 0xD, ft, imm16);
750}
751
752void MipsAssembler::EmitBcondR2(BranchCondition cond, Register rs, Register rt, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200753 switch (cond) {
754 case kCondLTZ:
755 CHECK_EQ(rt, ZERO);
756 Bltz(rs, imm16);
757 break;
758 case kCondGEZ:
759 CHECK_EQ(rt, ZERO);
760 Bgez(rs, imm16);
761 break;
762 case kCondLEZ:
763 CHECK_EQ(rt, ZERO);
764 Blez(rs, imm16);
765 break;
766 case kCondGTZ:
767 CHECK_EQ(rt, ZERO);
768 Bgtz(rs, imm16);
769 break;
770 case kCondEQ:
771 Beq(rs, rt, imm16);
772 break;
773 case kCondNE:
774 Bne(rs, rt, imm16);
775 break;
776 case kCondEQZ:
777 CHECK_EQ(rt, ZERO);
778 Beqz(rs, imm16);
779 break;
780 case kCondNEZ:
781 CHECK_EQ(rt, ZERO);
782 Bnez(rs, imm16);
783 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800784 case kCondF:
785 CHECK_EQ(rt, ZERO);
786 Bc1f(static_cast<int>(rs), imm16);
787 break;
788 case kCondT:
789 CHECK_EQ(rt, ZERO);
790 Bc1t(static_cast<int>(rs), imm16);
791 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200792 case kCondLT:
793 case kCondGE:
794 case kCondLE:
795 case kCondGT:
796 case kCondLTU:
797 case kCondGEU:
798 case kUncond:
799 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
800 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
801 LOG(FATAL) << "Unexpected branch condition " << cond;
802 UNREACHABLE();
803 }
804}
805
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800806void MipsAssembler::EmitBcondR6(BranchCondition cond, Register rs, Register rt, uint32_t imm16_21) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200807 switch (cond) {
808 case kCondLT:
809 Bltc(rs, rt, imm16_21);
810 break;
811 case kCondGE:
812 Bgec(rs, rt, imm16_21);
813 break;
814 case kCondLE:
815 Bgec(rt, rs, imm16_21);
816 break;
817 case kCondGT:
818 Bltc(rt, rs, imm16_21);
819 break;
820 case kCondLTZ:
821 CHECK_EQ(rt, ZERO);
822 Bltzc(rs, imm16_21);
823 break;
824 case kCondGEZ:
825 CHECK_EQ(rt, ZERO);
826 Bgezc(rs, imm16_21);
827 break;
828 case kCondLEZ:
829 CHECK_EQ(rt, ZERO);
830 Blezc(rs, imm16_21);
831 break;
832 case kCondGTZ:
833 CHECK_EQ(rt, ZERO);
834 Bgtzc(rs, imm16_21);
835 break;
836 case kCondEQ:
837 Beqc(rs, rt, imm16_21);
838 break;
839 case kCondNE:
840 Bnec(rs, rt, imm16_21);
841 break;
842 case kCondEQZ:
843 CHECK_EQ(rt, ZERO);
844 Beqzc(rs, imm16_21);
845 break;
846 case kCondNEZ:
847 CHECK_EQ(rt, ZERO);
848 Bnezc(rs, imm16_21);
849 break;
850 case kCondLTU:
851 Bltuc(rs, rt, imm16_21);
852 break;
853 case kCondGEU:
854 Bgeuc(rs, rt, imm16_21);
855 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800856 case kCondF:
857 CHECK_EQ(rt, ZERO);
858 Bc1eqz(static_cast<FRegister>(rs), imm16_21);
859 break;
860 case kCondT:
861 CHECK_EQ(rt, ZERO);
862 Bc1nez(static_cast<FRegister>(rs), imm16_21);
863 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200864 case kUncond:
865 LOG(FATAL) << "Unexpected branch condition " << cond;
866 UNREACHABLE();
867 }
jeffhao7fbee072012-08-24 17:56:54 -0700868}
869
870void MipsAssembler::AddS(FRegister fd, FRegister fs, FRegister ft) {
871 EmitFR(0x11, 0x10, ft, fs, fd, 0x0);
872}
873
874void MipsAssembler::SubS(FRegister fd, FRegister fs, FRegister ft) {
875 EmitFR(0x11, 0x10, ft, fs, fd, 0x1);
876}
877
878void MipsAssembler::MulS(FRegister fd, FRegister fs, FRegister ft) {
879 EmitFR(0x11, 0x10, ft, fs, fd, 0x2);
880}
881
882void MipsAssembler::DivS(FRegister fd, FRegister fs, FRegister ft) {
883 EmitFR(0x11, 0x10, ft, fs, fd, 0x3);
884}
885
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200886void MipsAssembler::AddD(FRegister fd, FRegister fs, FRegister ft) {
887 EmitFR(0x11, 0x11, ft, fs, fd, 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700888}
889
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200890void MipsAssembler::SubD(FRegister fd, FRegister fs, FRegister ft) {
891 EmitFR(0x11, 0x11, ft, fs, fd, 0x1);
jeffhao7fbee072012-08-24 17:56:54 -0700892}
893
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200894void MipsAssembler::MulD(FRegister fd, FRegister fs, FRegister ft) {
895 EmitFR(0x11, 0x11, ft, fs, fd, 0x2);
jeffhao7fbee072012-08-24 17:56:54 -0700896}
897
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200898void MipsAssembler::DivD(FRegister fd, FRegister fs, FRegister ft) {
899 EmitFR(0x11, 0x11, ft, fs, fd, 0x3);
jeffhao7fbee072012-08-24 17:56:54 -0700900}
901
Chris Larsenb74353a2015-11-20 09:07:09 -0800902void MipsAssembler::SqrtS(FRegister fd, FRegister fs) {
903 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x4);
904}
905
906void MipsAssembler::SqrtD(FRegister fd, FRegister fs) {
907 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x4);
908}
909
910void MipsAssembler::AbsS(FRegister fd, FRegister fs) {
911 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x5);
912}
913
914void MipsAssembler::AbsD(FRegister fd, FRegister fs) {
915 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x5);
916}
917
jeffhao7fbee072012-08-24 17:56:54 -0700918void MipsAssembler::MovS(FRegister fd, FRegister fs) {
919 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x6);
920}
921
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200922void MipsAssembler::MovD(FRegister fd, FRegister fs) {
923 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x6);
924}
925
926void MipsAssembler::NegS(FRegister fd, FRegister fs) {
927 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x7);
928}
929
930void MipsAssembler::NegD(FRegister fd, FRegister fs) {
931 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x7);
932}
933
Chris Larsenb74353a2015-11-20 09:07:09 -0800934void MipsAssembler::CunS(FRegister fs, FRegister ft) {
935 CunS(0, fs, ft);
936}
937
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800938void MipsAssembler::CunS(int cc, FRegister fs, FRegister ft) {
939 CHECK(!IsR6());
940 CHECK(IsUint<3>(cc)) << cc;
941 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x31);
942}
943
Chris Larsenb74353a2015-11-20 09:07:09 -0800944void MipsAssembler::CeqS(FRegister fs, FRegister ft) {
945 CeqS(0, fs, ft);
946}
947
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800948void MipsAssembler::CeqS(int cc, FRegister fs, FRegister ft) {
949 CHECK(!IsR6());
950 CHECK(IsUint<3>(cc)) << cc;
951 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x32);
952}
953
Chris Larsenb74353a2015-11-20 09:07:09 -0800954void MipsAssembler::CueqS(FRegister fs, FRegister ft) {
955 CueqS(0, fs, ft);
956}
957
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800958void MipsAssembler::CueqS(int cc, FRegister fs, FRegister ft) {
959 CHECK(!IsR6());
960 CHECK(IsUint<3>(cc)) << cc;
961 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x33);
962}
963
Chris Larsenb74353a2015-11-20 09:07:09 -0800964void MipsAssembler::ColtS(FRegister fs, FRegister ft) {
965 ColtS(0, fs, ft);
966}
967
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800968void MipsAssembler::ColtS(int cc, FRegister fs, FRegister ft) {
969 CHECK(!IsR6());
970 CHECK(IsUint<3>(cc)) << cc;
971 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x34);
972}
973
Chris Larsenb74353a2015-11-20 09:07:09 -0800974void MipsAssembler::CultS(FRegister fs, FRegister ft) {
975 CultS(0, fs, ft);
976}
977
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800978void MipsAssembler::CultS(int cc, FRegister fs, FRegister ft) {
979 CHECK(!IsR6());
980 CHECK(IsUint<3>(cc)) << cc;
981 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x35);
982}
983
Chris Larsenb74353a2015-11-20 09:07:09 -0800984void MipsAssembler::ColeS(FRegister fs, FRegister ft) {
985 ColeS(0, fs, ft);
986}
987
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800988void MipsAssembler::ColeS(int cc, FRegister fs, FRegister ft) {
989 CHECK(!IsR6());
990 CHECK(IsUint<3>(cc)) << cc;
991 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x36);
992}
993
Chris Larsenb74353a2015-11-20 09:07:09 -0800994void MipsAssembler::CuleS(FRegister fs, FRegister ft) {
995 CuleS(0, fs, ft);
996}
997
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800998void MipsAssembler::CuleS(int cc, FRegister fs, FRegister ft) {
999 CHECK(!IsR6());
1000 CHECK(IsUint<3>(cc)) << cc;
1001 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x37);
1002}
1003
Chris Larsenb74353a2015-11-20 09:07:09 -08001004void MipsAssembler::CunD(FRegister fs, FRegister ft) {
1005 CunD(0, fs, ft);
1006}
1007
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001008void MipsAssembler::CunD(int cc, FRegister fs, FRegister ft) {
1009 CHECK(!IsR6());
1010 CHECK(IsUint<3>(cc)) << cc;
1011 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x31);
1012}
1013
Chris Larsenb74353a2015-11-20 09:07:09 -08001014void MipsAssembler::CeqD(FRegister fs, FRegister ft) {
1015 CeqD(0, fs, ft);
1016}
1017
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001018void MipsAssembler::CeqD(int cc, FRegister fs, FRegister ft) {
1019 CHECK(!IsR6());
1020 CHECK(IsUint<3>(cc)) << cc;
1021 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x32);
1022}
1023
Chris Larsenb74353a2015-11-20 09:07:09 -08001024void MipsAssembler::CueqD(FRegister fs, FRegister ft) {
1025 CueqD(0, fs, ft);
1026}
1027
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001028void MipsAssembler::CueqD(int cc, FRegister fs, FRegister ft) {
1029 CHECK(!IsR6());
1030 CHECK(IsUint<3>(cc)) << cc;
1031 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x33);
1032}
1033
Chris Larsenb74353a2015-11-20 09:07:09 -08001034void MipsAssembler::ColtD(FRegister fs, FRegister ft) {
1035 ColtD(0, fs, ft);
1036}
1037
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001038void MipsAssembler::ColtD(int cc, FRegister fs, FRegister ft) {
1039 CHECK(!IsR6());
1040 CHECK(IsUint<3>(cc)) << cc;
1041 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x34);
1042}
1043
Chris Larsenb74353a2015-11-20 09:07:09 -08001044void MipsAssembler::CultD(FRegister fs, FRegister ft) {
1045 CultD(0, fs, ft);
1046}
1047
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001048void MipsAssembler::CultD(int cc, FRegister fs, FRegister ft) {
1049 CHECK(!IsR6());
1050 CHECK(IsUint<3>(cc)) << cc;
1051 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x35);
1052}
1053
Chris Larsenb74353a2015-11-20 09:07:09 -08001054void MipsAssembler::ColeD(FRegister fs, FRegister ft) {
1055 ColeD(0, fs, ft);
1056}
1057
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001058void MipsAssembler::ColeD(int cc, FRegister fs, FRegister ft) {
1059 CHECK(!IsR6());
1060 CHECK(IsUint<3>(cc)) << cc;
1061 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x36);
1062}
1063
Chris Larsenb74353a2015-11-20 09:07:09 -08001064void MipsAssembler::CuleD(FRegister fs, FRegister ft) {
1065 CuleD(0, fs, ft);
1066}
1067
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001068void MipsAssembler::CuleD(int cc, FRegister fs, FRegister ft) {
1069 CHECK(!IsR6());
1070 CHECK(IsUint<3>(cc)) << cc;
1071 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x37);
1072}
1073
1074void MipsAssembler::CmpUnS(FRegister fd, FRegister fs, FRegister ft) {
1075 CHECK(IsR6());
1076 EmitFR(0x11, 0x14, ft, fs, fd, 0x01);
1077}
1078
1079void MipsAssembler::CmpEqS(FRegister fd, FRegister fs, FRegister ft) {
1080 CHECK(IsR6());
1081 EmitFR(0x11, 0x14, ft, fs, fd, 0x02);
1082}
1083
1084void MipsAssembler::CmpUeqS(FRegister fd, FRegister fs, FRegister ft) {
1085 CHECK(IsR6());
1086 EmitFR(0x11, 0x14, ft, fs, fd, 0x03);
1087}
1088
1089void MipsAssembler::CmpLtS(FRegister fd, FRegister fs, FRegister ft) {
1090 CHECK(IsR6());
1091 EmitFR(0x11, 0x14, ft, fs, fd, 0x04);
1092}
1093
1094void MipsAssembler::CmpUltS(FRegister fd, FRegister fs, FRegister ft) {
1095 CHECK(IsR6());
1096 EmitFR(0x11, 0x14, ft, fs, fd, 0x05);
1097}
1098
1099void MipsAssembler::CmpLeS(FRegister fd, FRegister fs, FRegister ft) {
1100 CHECK(IsR6());
1101 EmitFR(0x11, 0x14, ft, fs, fd, 0x06);
1102}
1103
1104void MipsAssembler::CmpUleS(FRegister fd, FRegister fs, FRegister ft) {
1105 CHECK(IsR6());
1106 EmitFR(0x11, 0x14, ft, fs, fd, 0x07);
1107}
1108
1109void MipsAssembler::CmpOrS(FRegister fd, FRegister fs, FRegister ft) {
1110 CHECK(IsR6());
1111 EmitFR(0x11, 0x14, ft, fs, fd, 0x11);
1112}
1113
1114void MipsAssembler::CmpUneS(FRegister fd, FRegister fs, FRegister ft) {
1115 CHECK(IsR6());
1116 EmitFR(0x11, 0x14, ft, fs, fd, 0x12);
1117}
1118
1119void MipsAssembler::CmpNeS(FRegister fd, FRegister fs, FRegister ft) {
1120 CHECK(IsR6());
1121 EmitFR(0x11, 0x14, ft, fs, fd, 0x13);
1122}
1123
1124void MipsAssembler::CmpUnD(FRegister fd, FRegister fs, FRegister ft) {
1125 CHECK(IsR6());
1126 EmitFR(0x11, 0x15, ft, fs, fd, 0x01);
1127}
1128
1129void MipsAssembler::CmpEqD(FRegister fd, FRegister fs, FRegister ft) {
1130 CHECK(IsR6());
1131 EmitFR(0x11, 0x15, ft, fs, fd, 0x02);
1132}
1133
1134void MipsAssembler::CmpUeqD(FRegister fd, FRegister fs, FRegister ft) {
1135 CHECK(IsR6());
1136 EmitFR(0x11, 0x15, ft, fs, fd, 0x03);
1137}
1138
1139void MipsAssembler::CmpLtD(FRegister fd, FRegister fs, FRegister ft) {
1140 CHECK(IsR6());
1141 EmitFR(0x11, 0x15, ft, fs, fd, 0x04);
1142}
1143
1144void MipsAssembler::CmpUltD(FRegister fd, FRegister fs, FRegister ft) {
1145 CHECK(IsR6());
1146 EmitFR(0x11, 0x15, ft, fs, fd, 0x05);
1147}
1148
1149void MipsAssembler::CmpLeD(FRegister fd, FRegister fs, FRegister ft) {
1150 CHECK(IsR6());
1151 EmitFR(0x11, 0x15, ft, fs, fd, 0x06);
1152}
1153
1154void MipsAssembler::CmpUleD(FRegister fd, FRegister fs, FRegister ft) {
1155 CHECK(IsR6());
1156 EmitFR(0x11, 0x15, ft, fs, fd, 0x07);
1157}
1158
1159void MipsAssembler::CmpOrD(FRegister fd, FRegister fs, FRegister ft) {
1160 CHECK(IsR6());
1161 EmitFR(0x11, 0x15, ft, fs, fd, 0x11);
1162}
1163
1164void MipsAssembler::CmpUneD(FRegister fd, FRegister fs, FRegister ft) {
1165 CHECK(IsR6());
1166 EmitFR(0x11, 0x15, ft, fs, fd, 0x12);
1167}
1168
1169void MipsAssembler::CmpNeD(FRegister fd, FRegister fs, FRegister ft) {
1170 CHECK(IsR6());
1171 EmitFR(0x11, 0x15, ft, fs, fd, 0x13);
1172}
1173
1174void MipsAssembler::Movf(Register rd, Register rs, int cc) {
1175 CHECK(!IsR6());
1176 CHECK(IsUint<3>(cc)) << cc;
1177 EmitR(0, rs, static_cast<Register>(cc << 2), rd, 0, 0x01);
1178}
1179
1180void MipsAssembler::Movt(Register rd, Register rs, int cc) {
1181 CHECK(!IsR6());
1182 CHECK(IsUint<3>(cc)) << cc;
1183 EmitR(0, rs, static_cast<Register>((cc << 2) | 1), rd, 0, 0x01);
1184}
1185
Chris Larsenb74353a2015-11-20 09:07:09 -08001186void MipsAssembler::MovfS(FRegister fd, FRegister fs, int cc) {
1187 CHECK(!IsR6());
1188 CHECK(IsUint<3>(cc)) << cc;
1189 EmitFR(0x11, 0x10, static_cast<FRegister>(cc << 2), fs, fd, 0x11);
1190}
1191
1192void MipsAssembler::MovfD(FRegister fd, FRegister fs, int cc) {
1193 CHECK(!IsR6());
1194 CHECK(IsUint<3>(cc)) << cc;
1195 EmitFR(0x11, 0x11, static_cast<FRegister>(cc << 2), fs, fd, 0x11);
1196}
1197
1198void MipsAssembler::MovtS(FRegister fd, FRegister fs, int cc) {
1199 CHECK(!IsR6());
1200 CHECK(IsUint<3>(cc)) << cc;
1201 EmitFR(0x11, 0x10, static_cast<FRegister>((cc << 2) | 1), fs, fd, 0x11);
1202}
1203
1204void MipsAssembler::MovtD(FRegister fd, FRegister fs, int cc) {
1205 CHECK(!IsR6());
1206 CHECK(IsUint<3>(cc)) << cc;
1207 EmitFR(0x11, 0x11, static_cast<FRegister>((cc << 2) | 1), fs, fd, 0x11);
1208}
1209
1210void MipsAssembler::SelS(FRegister fd, FRegister fs, FRegister ft) {
1211 CHECK(IsR6());
1212 EmitFR(0x11, 0x10, ft, fs, fd, 0x10);
1213}
1214
1215void MipsAssembler::SelD(FRegister fd, FRegister fs, FRegister ft) {
1216 CHECK(IsR6());
1217 EmitFR(0x11, 0x11, ft, fs, fd, 0x10);
1218}
1219
1220void MipsAssembler::ClassS(FRegister fd, FRegister fs) {
1221 CHECK(IsR6());
1222 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x1b);
1223}
1224
1225void MipsAssembler::ClassD(FRegister fd, FRegister fs) {
1226 CHECK(IsR6());
1227 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x1b);
1228}
1229
1230void MipsAssembler::MinS(FRegister fd, FRegister fs, FRegister ft) {
1231 CHECK(IsR6());
1232 EmitFR(0x11, 0x10, ft, fs, fd, 0x1c);
1233}
1234
1235void MipsAssembler::MinD(FRegister fd, FRegister fs, FRegister ft) {
1236 CHECK(IsR6());
1237 EmitFR(0x11, 0x11, ft, fs, fd, 0x1c);
1238}
1239
1240void MipsAssembler::MaxS(FRegister fd, FRegister fs, FRegister ft) {
1241 CHECK(IsR6());
1242 EmitFR(0x11, 0x10, ft, fs, fd, 0x1e);
1243}
1244
1245void MipsAssembler::MaxD(FRegister fd, FRegister fs, FRegister ft) {
1246 CHECK(IsR6());
1247 EmitFR(0x11, 0x11, ft, fs, fd, 0x1e);
1248}
1249
Alexey Frunzebaf60b72015-12-22 15:15:03 -08001250void MipsAssembler::TruncLS(FRegister fd, FRegister fs) {
1251 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x09);
1252}
1253
1254void MipsAssembler::TruncLD(FRegister fd, FRegister fs) {
1255 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x09);
1256}
1257
1258void MipsAssembler::TruncWS(FRegister fd, FRegister fs) {
1259 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x0D);
1260}
1261
1262void MipsAssembler::TruncWD(FRegister fd, FRegister fs) {
1263 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x0D);
1264}
1265
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001266void MipsAssembler::Cvtsw(FRegister fd, FRegister fs) {
1267 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x20);
1268}
1269
1270void MipsAssembler::Cvtdw(FRegister fd, FRegister fs) {
1271 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x21);
1272}
1273
1274void MipsAssembler::Cvtsd(FRegister fd, FRegister fs) {
1275 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x20);
1276}
1277
1278void MipsAssembler::Cvtds(FRegister fd, FRegister fs) {
1279 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x21);
jeffhao7fbee072012-08-24 17:56:54 -07001280}
1281
Alexey Frunzebaf60b72015-12-22 15:15:03 -08001282void MipsAssembler::Cvtsl(FRegister fd, FRegister fs) {
1283 EmitFR(0x11, 0x15, static_cast<FRegister>(0), fs, fd, 0x20);
1284}
1285
1286void MipsAssembler::Cvtdl(FRegister fd, FRegister fs) {
1287 EmitFR(0x11, 0x15, static_cast<FRegister>(0), fs, fd, 0x21);
1288}
1289
Chris Larsenb74353a2015-11-20 09:07:09 -08001290void MipsAssembler::FloorWS(FRegister fd, FRegister fs) {
1291 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0xf);
1292}
1293
1294void MipsAssembler::FloorWD(FRegister fd, FRegister fs) {
1295 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0xf);
1296}
1297
jeffhao7fbee072012-08-24 17:56:54 -07001298void MipsAssembler::Mfc1(Register rt, FRegister fs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001299 EmitFR(0x11, 0x00, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -07001300}
1301
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001302void MipsAssembler::Mtc1(Register rt, FRegister fs) {
1303 EmitFR(0x11, 0x04, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
1304}
1305
1306void MipsAssembler::Mfhc1(Register rt, FRegister fs) {
1307 EmitFR(0x11, 0x03, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
1308}
1309
1310void MipsAssembler::Mthc1(Register rt, FRegister fs) {
1311 EmitFR(0x11, 0x07, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -07001312}
1313
Alexey Frunzebb9863a2016-01-11 15:51:16 -08001314void MipsAssembler::MoveFromFpuHigh(Register rt, FRegister fs) {
1315 if (Is32BitFPU()) {
1316 CHECK_EQ(fs % 2, 0) << fs;
1317 Mfc1(rt, static_cast<FRegister>(fs + 1));
1318 } else {
1319 Mfhc1(rt, fs);
1320 }
1321}
1322
1323void MipsAssembler::MoveToFpuHigh(Register rt, FRegister fs) {
1324 if (Is32BitFPU()) {
1325 CHECK_EQ(fs % 2, 0) << fs;
1326 Mtc1(rt, static_cast<FRegister>(fs + 1));
1327 } else {
1328 Mthc1(rt, fs);
1329 }
1330}
1331
jeffhao7fbee072012-08-24 17:56:54 -07001332void MipsAssembler::Lwc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001333 EmitI(0x31, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001334}
1335
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001336void MipsAssembler::Ldc1(FRegister ft, Register rs, uint16_t imm16) {
1337 EmitI(0x35, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001338}
1339
1340void MipsAssembler::Swc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001341 EmitI(0x39, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001342}
1343
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001344void MipsAssembler::Sdc1(FRegister ft, Register rs, uint16_t imm16) {
1345 EmitI(0x3d, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001346}
1347
1348void MipsAssembler::Break() {
1349 EmitR(0, static_cast<Register>(0), static_cast<Register>(0),
1350 static_cast<Register>(0), 0, 0xD);
1351}
1352
jeffhao07030602012-09-26 14:33:14 -07001353void MipsAssembler::Nop() {
1354 EmitR(0x0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0), 0, 0x0);
1355}
1356
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001357void MipsAssembler::Move(Register rd, Register rs) {
1358 Or(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001359}
1360
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001361void MipsAssembler::Clear(Register rd) {
1362 Move(rd, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001363}
1364
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001365void MipsAssembler::Not(Register rd, Register rs) {
1366 Nor(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001367}
1368
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001369void MipsAssembler::Push(Register rs) {
1370 IncreaseFrameSize(kMipsWordSize);
1371 Sw(rs, SP, 0);
jeffhao7fbee072012-08-24 17:56:54 -07001372}
1373
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001374void MipsAssembler::Pop(Register rd) {
1375 Lw(rd, SP, 0);
1376 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001377}
1378
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001379void MipsAssembler::PopAndReturn(Register rd, Register rt) {
1380 Lw(rd, SP, 0);
1381 Jr(rt);
1382 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001383}
1384
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001385void MipsAssembler::LoadConst32(Register rd, int32_t value) {
1386 if (IsUint<16>(value)) {
1387 // Use OR with (unsigned) immediate to encode 16b unsigned int.
1388 Ori(rd, ZERO, value);
1389 } else if (IsInt<16>(value)) {
1390 // Use ADD with (signed) immediate to encode 16b signed int.
1391 Addiu(rd, ZERO, value);
jeffhao7fbee072012-08-24 17:56:54 -07001392 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001393 Lui(rd, High16Bits(value));
1394 if (value & 0xFFFF)
1395 Ori(rd, rd, Low16Bits(value));
1396 }
1397}
1398
1399void MipsAssembler::LoadConst64(Register reg_hi, Register reg_lo, int64_t value) {
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001400 uint32_t low = Low32Bits(value);
1401 uint32_t high = High32Bits(value);
1402 LoadConst32(reg_lo, low);
1403 if (high != low) {
1404 LoadConst32(reg_hi, high);
1405 } else {
1406 Move(reg_hi, reg_lo);
1407 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001408}
1409
1410void MipsAssembler::StoreConst32ToOffset(int32_t value,
1411 Register base,
1412 int32_t offset,
1413 Register temp) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07001414 CHECK_NE(temp, AT); // Must not use AT as temp, so as not to overwrite the adjusted base.
1415 AdjustBaseAndOffset(base, offset, /* is_doubleword */ false);
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001416 if (value == 0) {
1417 temp = ZERO;
1418 } else {
1419 LoadConst32(temp, value);
1420 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001421 Sw(temp, base, offset);
1422}
1423
1424void MipsAssembler::StoreConst64ToOffset(int64_t value,
1425 Register base,
1426 int32_t offset,
1427 Register temp) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07001428 CHECK_NE(temp, AT); // Must not use AT as temp, so as not to overwrite the adjusted base.
1429 AdjustBaseAndOffset(base, offset, /* is_doubleword */ true);
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001430 uint32_t low = Low32Bits(value);
1431 uint32_t high = High32Bits(value);
1432 if (low == 0) {
1433 Sw(ZERO, base, offset);
1434 } else {
1435 LoadConst32(temp, low);
1436 Sw(temp, base, offset);
1437 }
1438 if (high == 0) {
1439 Sw(ZERO, base, offset + kMipsWordSize);
1440 } else {
1441 if (high != low) {
1442 LoadConst32(temp, high);
1443 }
1444 Sw(temp, base, offset + kMipsWordSize);
1445 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001446}
1447
1448void MipsAssembler::LoadSConst32(FRegister r, int32_t value, Register temp) {
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001449 if (value == 0) {
1450 temp = ZERO;
1451 } else {
1452 LoadConst32(temp, value);
1453 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001454 Mtc1(temp, r);
1455}
1456
1457void MipsAssembler::LoadDConst64(FRegister rd, int64_t value, Register temp) {
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001458 uint32_t low = Low32Bits(value);
1459 uint32_t high = High32Bits(value);
1460 if (low == 0) {
1461 Mtc1(ZERO, rd);
1462 } else {
1463 LoadConst32(temp, low);
1464 Mtc1(temp, rd);
1465 }
1466 if (high == 0) {
Alexey Frunzebb9863a2016-01-11 15:51:16 -08001467 MoveToFpuHigh(ZERO, rd);
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001468 } else {
1469 LoadConst32(temp, high);
Alexey Frunzebb9863a2016-01-11 15:51:16 -08001470 MoveToFpuHigh(temp, rd);
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001471 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001472}
1473
1474void MipsAssembler::Addiu32(Register rt, Register rs, int32_t value, Register temp) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07001475 CHECK_NE(rs, temp); // Must not overwrite the register `rs` while loading `value`.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001476 if (IsInt<16>(value)) {
1477 Addiu(rt, rs, value);
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07001478 } else if (IsR6()) {
1479 int16_t high = High16Bits(value);
1480 int16_t low = Low16Bits(value);
1481 high += (low < 0) ? 1 : 0; // Account for sign extension in addiu.
1482 if (low != 0) {
1483 Aui(temp, rs, high);
1484 Addiu(rt, temp, low);
1485 } else {
1486 Aui(rt, rs, high);
1487 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001488 } else {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07001489 // Do not load the whole 32-bit `value` if it can be represented as
1490 // a sum of two 16-bit signed values. This can save an instruction.
1491 constexpr int32_t kMinValueForSimpleAdjustment = std::numeric_limits<int16_t>::min() * 2;
1492 constexpr int32_t kMaxValueForSimpleAdjustment = std::numeric_limits<int16_t>::max() * 2;
1493 if (0 <= value && value <= kMaxValueForSimpleAdjustment) {
1494 Addiu(temp, rs, kMaxValueForSimpleAdjustment / 2);
1495 Addiu(rt, temp, value - kMaxValueForSimpleAdjustment / 2);
1496 } else if (kMinValueForSimpleAdjustment <= value && value < 0) {
1497 Addiu(temp, rs, kMinValueForSimpleAdjustment / 2);
1498 Addiu(rt, temp, value - kMinValueForSimpleAdjustment / 2);
1499 } else {
1500 // Now that all shorter options have been exhausted, load the full 32-bit value.
1501 LoadConst32(temp, value);
1502 Addu(rt, rs, temp);
1503 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001504 }
1505}
1506
1507void MipsAssembler::Branch::InitShortOrLong(MipsAssembler::Branch::OffsetBits offset_size,
1508 MipsAssembler::Branch::Type short_type,
1509 MipsAssembler::Branch::Type long_type) {
1510 type_ = (offset_size <= branch_info_[short_type].offset_size) ? short_type : long_type;
1511}
1512
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001513void MipsAssembler::Branch::InitializeType(bool is_call, bool is_literal, bool is_r6) {
1514 CHECK_EQ(is_call && is_literal, false);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001515 OffsetBits offset_size = GetOffsetSizeNeeded(location_, target_);
1516 if (is_r6) {
1517 // R6
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001518 if (is_literal) {
1519 CHECK(!IsResolved());
1520 type_ = kR6Literal;
1521 } else if (is_call) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001522 InitShortOrLong(offset_size, kR6Call, kR6LongCall);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001523 } else {
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001524 switch (condition_) {
1525 case kUncond:
1526 InitShortOrLong(offset_size, kR6UncondBranch, kR6LongUncondBranch);
1527 break;
1528 case kCondEQZ:
1529 case kCondNEZ:
1530 // Special case for beqzc/bnezc with longer offset than in other b<cond>c instructions.
1531 type_ = (offset_size <= kOffset23) ? kR6CondBranch : kR6LongCondBranch;
1532 break;
1533 default:
1534 InitShortOrLong(offset_size, kR6CondBranch, kR6LongCondBranch);
1535 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001536 }
1537 }
1538 } else {
1539 // R2
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001540 if (is_literal) {
1541 CHECK(!IsResolved());
1542 type_ = kLiteral;
1543 } else if (is_call) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001544 InitShortOrLong(offset_size, kCall, kLongCall);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001545 } else {
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001546 switch (condition_) {
1547 case kUncond:
1548 InitShortOrLong(offset_size, kUncondBranch, kLongUncondBranch);
1549 break;
1550 default:
1551 InitShortOrLong(offset_size, kCondBranch, kLongCondBranch);
1552 break;
1553 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001554 }
1555 }
1556 old_type_ = type_;
1557}
1558
1559bool MipsAssembler::Branch::IsNop(BranchCondition condition, Register lhs, Register rhs) {
1560 switch (condition) {
1561 case kCondLT:
1562 case kCondGT:
1563 case kCondNE:
1564 case kCondLTU:
1565 return lhs == rhs;
1566 default:
1567 return false;
1568 }
1569}
1570
1571bool MipsAssembler::Branch::IsUncond(BranchCondition condition, Register lhs, Register rhs) {
1572 switch (condition) {
1573 case kUncond:
1574 return true;
1575 case kCondGE:
1576 case kCondLE:
1577 case kCondEQ:
1578 case kCondGEU:
1579 return lhs == rhs;
1580 default:
1581 return false;
1582 }
1583}
1584
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001585MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target, bool is_call)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001586 : old_location_(location),
1587 location_(location),
1588 target_(target),
1589 lhs_reg_(0),
1590 rhs_reg_(0),
1591 condition_(kUncond) {
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001592 InitializeType(is_call, /* is_literal */ false, is_r6);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001593}
1594
1595MipsAssembler::Branch::Branch(bool is_r6,
1596 uint32_t location,
1597 uint32_t target,
1598 MipsAssembler::BranchCondition condition,
1599 Register lhs_reg,
1600 Register rhs_reg)
1601 : old_location_(location),
1602 location_(location),
1603 target_(target),
1604 lhs_reg_(lhs_reg),
1605 rhs_reg_(rhs_reg),
1606 condition_(condition) {
1607 CHECK_NE(condition, kUncond);
1608 switch (condition) {
1609 case kCondLT:
1610 case kCondGE:
1611 case kCondLE:
1612 case kCondGT:
1613 case kCondLTU:
1614 case kCondGEU:
1615 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
1616 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
1617 // We leave this up to the caller.
1618 CHECK(is_r6);
1619 FALLTHROUGH_INTENDED;
1620 case kCondEQ:
1621 case kCondNE:
1622 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1623 // To compare with 0, use dedicated kCond*Z conditions.
1624 CHECK_NE(lhs_reg, ZERO);
1625 CHECK_NE(rhs_reg, ZERO);
1626 break;
1627 case kCondLTZ:
1628 case kCondGEZ:
1629 case kCondLEZ:
1630 case kCondGTZ:
1631 case kCondEQZ:
1632 case kCondNEZ:
1633 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1634 CHECK_NE(lhs_reg, ZERO);
1635 CHECK_EQ(rhs_reg, ZERO);
1636 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001637 case kCondF:
1638 case kCondT:
1639 CHECK_EQ(rhs_reg, ZERO);
1640 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001641 case kUncond:
1642 UNREACHABLE();
1643 }
1644 CHECK(!IsNop(condition, lhs_reg, rhs_reg));
1645 if (IsUncond(condition, lhs_reg, rhs_reg)) {
1646 // Branch condition is always true, make the branch unconditional.
1647 condition_ = kUncond;
1648 }
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001649 InitializeType(/* is_call */ false, /* is_literal */ false, is_r6);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001650}
1651
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001652MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, Register dest_reg, Register base_reg)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001653 : old_location_(location),
1654 location_(location),
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001655 target_(kUnresolved),
1656 lhs_reg_(dest_reg),
1657 rhs_reg_(base_reg),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001658 condition_(kUncond) {
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001659 CHECK_NE(dest_reg, ZERO);
1660 if (is_r6) {
1661 CHECK_EQ(base_reg, ZERO);
1662 } else {
1663 CHECK_NE(base_reg, ZERO);
1664 }
1665 InitializeType(/* is_call */ false, /* is_literal */ true, is_r6);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001666}
1667
1668MipsAssembler::BranchCondition MipsAssembler::Branch::OppositeCondition(
1669 MipsAssembler::BranchCondition cond) {
1670 switch (cond) {
1671 case kCondLT:
1672 return kCondGE;
1673 case kCondGE:
1674 return kCondLT;
1675 case kCondLE:
1676 return kCondGT;
1677 case kCondGT:
1678 return kCondLE;
1679 case kCondLTZ:
1680 return kCondGEZ;
1681 case kCondGEZ:
1682 return kCondLTZ;
1683 case kCondLEZ:
1684 return kCondGTZ;
1685 case kCondGTZ:
1686 return kCondLEZ;
1687 case kCondEQ:
1688 return kCondNE;
1689 case kCondNE:
1690 return kCondEQ;
1691 case kCondEQZ:
1692 return kCondNEZ;
1693 case kCondNEZ:
1694 return kCondEQZ;
1695 case kCondLTU:
1696 return kCondGEU;
1697 case kCondGEU:
1698 return kCondLTU;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001699 case kCondF:
1700 return kCondT;
1701 case kCondT:
1702 return kCondF;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001703 case kUncond:
1704 LOG(FATAL) << "Unexpected branch condition " << cond;
1705 }
1706 UNREACHABLE();
1707}
1708
1709MipsAssembler::Branch::Type MipsAssembler::Branch::GetType() const {
1710 return type_;
1711}
1712
1713MipsAssembler::BranchCondition MipsAssembler::Branch::GetCondition() const {
1714 return condition_;
1715}
1716
1717Register MipsAssembler::Branch::GetLeftRegister() const {
1718 return static_cast<Register>(lhs_reg_);
1719}
1720
1721Register MipsAssembler::Branch::GetRightRegister() const {
1722 return static_cast<Register>(rhs_reg_);
1723}
1724
1725uint32_t MipsAssembler::Branch::GetTarget() const {
1726 return target_;
1727}
1728
1729uint32_t MipsAssembler::Branch::GetLocation() const {
1730 return location_;
1731}
1732
1733uint32_t MipsAssembler::Branch::GetOldLocation() const {
1734 return old_location_;
1735}
1736
1737uint32_t MipsAssembler::Branch::GetLength() const {
1738 return branch_info_[type_].length;
1739}
1740
1741uint32_t MipsAssembler::Branch::GetOldLength() const {
1742 return branch_info_[old_type_].length;
1743}
1744
1745uint32_t MipsAssembler::Branch::GetSize() const {
1746 return GetLength() * sizeof(uint32_t);
1747}
1748
1749uint32_t MipsAssembler::Branch::GetOldSize() const {
1750 return GetOldLength() * sizeof(uint32_t);
1751}
1752
1753uint32_t MipsAssembler::Branch::GetEndLocation() const {
1754 return GetLocation() + GetSize();
1755}
1756
1757uint32_t MipsAssembler::Branch::GetOldEndLocation() const {
1758 return GetOldLocation() + GetOldSize();
1759}
1760
1761bool MipsAssembler::Branch::IsLong() const {
1762 switch (type_) {
1763 // R2 short branches.
1764 case kUncondBranch:
1765 case kCondBranch:
1766 case kCall:
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001767 // R2 near literal.
1768 case kLiteral:
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001769 // R6 short branches.
1770 case kR6UncondBranch:
1771 case kR6CondBranch:
1772 case kR6Call:
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001773 // R6 near literal.
1774 case kR6Literal:
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001775 return false;
1776 // R2 long branches.
1777 case kLongUncondBranch:
1778 case kLongCondBranch:
1779 case kLongCall:
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001780 // R2 far literal.
1781 case kFarLiteral:
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001782 // R6 long branches.
1783 case kR6LongUncondBranch:
1784 case kR6LongCondBranch:
1785 case kR6LongCall:
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001786 // R6 far literal.
1787 case kR6FarLiteral:
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001788 return true;
1789 }
1790 UNREACHABLE();
1791}
1792
1793bool MipsAssembler::Branch::IsResolved() const {
1794 return target_ != kUnresolved;
1795}
1796
1797MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSize() const {
1798 OffsetBits offset_size =
1799 (type_ == kR6CondBranch && (condition_ == kCondEQZ || condition_ == kCondNEZ))
1800 ? kOffset23
1801 : branch_info_[type_].offset_size;
1802 return offset_size;
1803}
1804
1805MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSizeNeeded(uint32_t location,
1806 uint32_t target) {
1807 // For unresolved targets assume the shortest encoding
1808 // (later it will be made longer if needed).
1809 if (target == kUnresolved)
1810 return kOffset16;
1811 int64_t distance = static_cast<int64_t>(target) - location;
1812 // To simplify calculations in composite branches consisting of multiple instructions
1813 // bump up the distance by a value larger than the max byte size of a composite branch.
1814 distance += (distance >= 0) ? kMaxBranchSize : -kMaxBranchSize;
1815 if (IsInt<kOffset16>(distance))
1816 return kOffset16;
1817 else if (IsInt<kOffset18>(distance))
1818 return kOffset18;
1819 else if (IsInt<kOffset21>(distance))
1820 return kOffset21;
1821 else if (IsInt<kOffset23>(distance))
1822 return kOffset23;
1823 else if (IsInt<kOffset28>(distance))
1824 return kOffset28;
1825 return kOffset32;
1826}
1827
1828void MipsAssembler::Branch::Resolve(uint32_t target) {
1829 target_ = target;
1830}
1831
1832void MipsAssembler::Branch::Relocate(uint32_t expand_location, uint32_t delta) {
1833 if (location_ > expand_location) {
1834 location_ += delta;
1835 }
1836 if (!IsResolved()) {
1837 return; // Don't know the target yet.
1838 }
1839 if (target_ > expand_location) {
1840 target_ += delta;
1841 }
1842}
1843
1844void MipsAssembler::Branch::PromoteToLong() {
1845 switch (type_) {
1846 // R2 short branches.
1847 case kUncondBranch:
1848 type_ = kLongUncondBranch;
1849 break;
1850 case kCondBranch:
1851 type_ = kLongCondBranch;
1852 break;
1853 case kCall:
1854 type_ = kLongCall;
1855 break;
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001856 // R2 near literal.
1857 case kLiteral:
1858 type_ = kFarLiteral;
1859 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001860 // R6 short branches.
1861 case kR6UncondBranch:
1862 type_ = kR6LongUncondBranch;
1863 break;
1864 case kR6CondBranch:
1865 type_ = kR6LongCondBranch;
1866 break;
1867 case kR6Call:
1868 type_ = kR6LongCall;
1869 break;
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001870 // R6 near literal.
1871 case kR6Literal:
1872 type_ = kR6FarLiteral;
1873 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001874 default:
1875 // Note: 'type_' is already long.
1876 break;
1877 }
1878 CHECK(IsLong());
1879}
1880
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001881uint32_t MipsAssembler::GetBranchLocationOrPcRelBase(const MipsAssembler::Branch* branch) const {
1882 switch (branch->GetType()) {
1883 case Branch::kLiteral:
1884 case Branch::kFarLiteral:
1885 return GetLabelLocation(&pc_rel_base_label_);
1886 default:
1887 return branch->GetLocation();
1888 }
1889}
1890
1891uint32_t MipsAssembler::Branch::PromoteIfNeeded(uint32_t location, uint32_t max_short_distance) {
1892 // `location` is either `GetLabelLocation(&pc_rel_base_label_)` for R2 literals or
1893 // `this->GetLocation()` for everything else.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001894 // If the branch is still unresolved or already long, nothing to do.
1895 if (IsLong() || !IsResolved()) {
1896 return 0;
1897 }
1898 // Promote the short branch to long if the offset size is too small
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001899 // to hold the distance between location and target_.
1900 if (GetOffsetSizeNeeded(location, target_) > GetOffsetSize()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001901 PromoteToLong();
1902 uint32_t old_size = GetOldSize();
1903 uint32_t new_size = GetSize();
1904 CHECK_GT(new_size, old_size);
1905 return new_size - old_size;
1906 }
1907 // The following logic is for debugging/testing purposes.
1908 // Promote some short branches to long when it's not really required.
1909 if (UNLIKELY(max_short_distance != std::numeric_limits<uint32_t>::max())) {
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001910 int64_t distance = static_cast<int64_t>(target_) - location;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001911 distance = (distance >= 0) ? distance : -distance;
1912 if (distance >= max_short_distance) {
1913 PromoteToLong();
1914 uint32_t old_size = GetOldSize();
1915 uint32_t new_size = GetSize();
1916 CHECK_GT(new_size, old_size);
1917 return new_size - old_size;
1918 }
1919 }
1920 return 0;
1921}
1922
1923uint32_t MipsAssembler::Branch::GetOffsetLocation() const {
1924 return location_ + branch_info_[type_].instr_offset * sizeof(uint32_t);
1925}
1926
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001927uint32_t MipsAssembler::GetBranchOrPcRelBaseForEncoding(const MipsAssembler::Branch* branch) const {
1928 switch (branch->GetType()) {
1929 case Branch::kLiteral:
1930 case Branch::kFarLiteral:
1931 return GetLabelLocation(&pc_rel_base_label_);
1932 default:
1933 return branch->GetOffsetLocation() +
1934 Branch::branch_info_[branch->GetType()].pc_org * sizeof(uint32_t);
1935 }
1936}
1937
1938uint32_t MipsAssembler::Branch::GetOffset(uint32_t location) const {
1939 // `location` is either `GetLabelLocation(&pc_rel_base_label_)` for R2 literals or
1940 // `this->GetOffsetLocation() + branch_info_[this->GetType()].pc_org * sizeof(uint32_t)`
1941 // for everything else.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001942 CHECK(IsResolved());
1943 uint32_t ofs_mask = 0xFFFFFFFF >> (32 - GetOffsetSize());
1944 // Calculate the byte distance between instructions and also account for
1945 // different PC-relative origins.
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001946 uint32_t offset = target_ - location;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001947 // Prepare the offset for encoding into the instruction(s).
1948 offset = (offset & ofs_mask) >> branch_info_[type_].offset_shift;
1949 return offset;
1950}
1951
1952MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) {
1953 CHECK_LT(branch_id, branches_.size());
1954 return &branches_[branch_id];
1955}
1956
1957const MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) const {
1958 CHECK_LT(branch_id, branches_.size());
1959 return &branches_[branch_id];
1960}
1961
1962void MipsAssembler::Bind(MipsLabel* label) {
1963 CHECK(!label->IsBound());
1964 uint32_t bound_pc = buffer_.Size();
1965
1966 // Walk the list of branches referring to and preceding this label.
1967 // Store the previously unknown target addresses in them.
1968 while (label->IsLinked()) {
1969 uint32_t branch_id = label->Position();
1970 Branch* branch = GetBranch(branch_id);
1971 branch->Resolve(bound_pc);
1972
1973 uint32_t branch_location = branch->GetLocation();
1974 // Extract the location of the previous branch in the list (walking the list backwards;
1975 // the previous branch ID was stored in the space reserved for this branch).
1976 uint32_t prev = buffer_.Load<uint32_t>(branch_location);
1977
1978 // On to the previous branch in the list...
1979 label->position_ = prev;
1980 }
1981
1982 // Now make the label object contain its own location (relative to the end of the preceding
1983 // branch, if any; it will be used by the branches referring to and following this label).
1984 label->prev_branch_id_plus_one_ = branches_.size();
1985 if (label->prev_branch_id_plus_one_) {
1986 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1987 const Branch* branch = GetBranch(branch_id);
1988 bound_pc -= branch->GetEndLocation();
1989 }
1990 label->BindTo(bound_pc);
1991}
1992
Alexey Frunzee3fb2452016-05-10 16:08:05 -07001993uint32_t MipsAssembler::GetLabelLocation(const MipsLabel* label) const {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001994 CHECK(label->IsBound());
1995 uint32_t target = label->Position();
1996 if (label->prev_branch_id_plus_one_) {
1997 // Get label location based on the branch preceding it.
1998 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1999 const Branch* branch = GetBranch(branch_id);
2000 target += branch->GetEndLocation();
2001 }
2002 return target;
2003}
2004
2005uint32_t MipsAssembler::GetAdjustedPosition(uint32_t old_position) {
2006 // We can reconstruct the adjustment by going through all the branches from the beginning
2007 // up to the old_position. Since we expect AdjustedPosition() to be called in a loop
2008 // with increasing old_position, we can use the data from last AdjustedPosition() to
2009 // continue where we left off and the whole loop should be O(m+n) where m is the number
2010 // of positions to adjust and n is the number of branches.
2011 if (old_position < last_old_position_) {
2012 last_position_adjustment_ = 0;
2013 last_old_position_ = 0;
2014 last_branch_id_ = 0;
2015 }
2016 while (last_branch_id_ != branches_.size()) {
2017 const Branch* branch = GetBranch(last_branch_id_);
2018 if (branch->GetLocation() >= old_position + last_position_adjustment_) {
2019 break;
2020 }
2021 last_position_adjustment_ += branch->GetSize() - branch->GetOldSize();
2022 ++last_branch_id_;
2023 }
2024 last_old_position_ = old_position;
2025 return old_position + last_position_adjustment_;
2026}
2027
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002028void MipsAssembler::BindPcRelBaseLabel() {
2029 Bind(&pc_rel_base_label_);
2030}
2031
Alexey Frunze06a46c42016-07-19 15:00:40 -07002032uint32_t MipsAssembler::GetPcRelBaseLabelLocation() const {
2033 return GetLabelLocation(&pc_rel_base_label_);
2034}
2035
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002036void MipsAssembler::FinalizeLabeledBranch(MipsLabel* label) {
2037 uint32_t length = branches_.back().GetLength();
2038 if (!label->IsBound()) {
2039 // Branch forward (to a following label), distance is unknown.
2040 // The first branch forward will contain 0, serving as the terminator of
2041 // the list of forward-reaching branches.
2042 Emit(label->position_);
2043 length--;
2044 // Now make the label object point to this branch
2045 // (this forms a linked list of branches preceding this label).
2046 uint32_t branch_id = branches_.size() - 1;
2047 label->LinkTo(branch_id);
2048 }
2049 // Reserve space for the branch.
2050 while (length--) {
2051 Nop();
2052 }
2053}
2054
2055void MipsAssembler::Buncond(MipsLabel* label) {
2056 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002057 branches_.emplace_back(IsR6(), buffer_.Size(), target, /* is_call */ false);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002058 FinalizeLabeledBranch(label);
2059}
2060
2061void MipsAssembler::Bcond(MipsLabel* label, BranchCondition condition, Register lhs, Register rhs) {
2062 // If lhs = rhs, this can be a NOP.
2063 if (Branch::IsNop(condition, lhs, rhs)) {
2064 return;
2065 }
2066 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
2067 branches_.emplace_back(IsR6(), buffer_.Size(), target, condition, lhs, rhs);
2068 FinalizeLabeledBranch(label);
2069}
2070
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002071void MipsAssembler::Call(MipsLabel* label) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002072 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002073 branches_.emplace_back(IsR6(), buffer_.Size(), target, /* is_call */ true);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002074 FinalizeLabeledBranch(label);
2075}
2076
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002077Literal* MipsAssembler::NewLiteral(size_t size, const uint8_t* data) {
2078 DCHECK(size == 4u || size == 8u) << size;
2079 literals_.emplace_back(size, data);
2080 return &literals_.back();
2081}
2082
2083void MipsAssembler::LoadLiteral(Register dest_reg, Register base_reg, Literal* literal) {
2084 // Literal loads are treated as pseudo branches since they require very similar handling.
2085 DCHECK_EQ(literal->GetSize(), 4u);
2086 MipsLabel* label = literal->GetLabel();
2087 DCHECK(!label->IsBound());
2088 branches_.emplace_back(IsR6(),
2089 buffer_.Size(),
2090 dest_reg,
2091 base_reg);
2092 FinalizeLabeledBranch(label);
2093}
2094
2095void MipsAssembler::EmitLiterals() {
2096 if (!literals_.empty()) {
2097 // We don't support byte and half-word literals.
2098 // TODO: proper alignment for 64-bit literals when they're implemented.
2099 for (Literal& literal : literals_) {
2100 MipsLabel* label = literal.GetLabel();
2101 Bind(label);
2102 AssemblerBuffer::EnsureCapacity ensured(&buffer_);
2103 DCHECK(literal.GetSize() == 4u || literal.GetSize() == 8u);
2104 for (size_t i = 0, size = literal.GetSize(); i != size; ++i) {
2105 buffer_.Emit<uint8_t>(literal.GetData()[i]);
2106 }
2107 }
2108 }
2109}
2110
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002111void MipsAssembler::PromoteBranches() {
2112 // Promote short branches to long as necessary.
2113 bool changed;
2114 do {
2115 changed = false;
2116 for (auto& branch : branches_) {
2117 CHECK(branch.IsResolved());
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002118 uint32_t base = GetBranchLocationOrPcRelBase(&branch);
2119 uint32_t delta = branch.PromoteIfNeeded(base);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002120 // If this branch has been promoted and needs to expand in size,
2121 // relocate all branches by the expansion size.
2122 if (delta) {
2123 changed = true;
2124 uint32_t expand_location = branch.GetLocation();
2125 for (auto& branch2 : branches_) {
2126 branch2.Relocate(expand_location, delta);
2127 }
2128 }
2129 }
2130 } while (changed);
2131
2132 // Account for branch expansion by resizing the code buffer
2133 // and moving the code in it to its final location.
2134 size_t branch_count = branches_.size();
2135 if (branch_count > 0) {
2136 // Resize.
2137 Branch& last_branch = branches_[branch_count - 1];
2138 uint32_t size_delta = last_branch.GetEndLocation() - last_branch.GetOldEndLocation();
2139 uint32_t old_size = buffer_.Size();
2140 buffer_.Resize(old_size + size_delta);
2141 // Move the code residing between branch placeholders.
2142 uint32_t end = old_size;
2143 for (size_t i = branch_count; i > 0; ) {
2144 Branch& branch = branches_[--i];
2145 uint32_t size = end - branch.GetOldEndLocation();
2146 buffer_.Move(branch.GetEndLocation(), branch.GetOldEndLocation(), size);
2147 end = branch.GetOldLocation();
2148 }
2149 }
2150}
2151
2152// Note: make sure branch_info_[] and EmitBranch() are kept synchronized.
2153const MipsAssembler::Branch::BranchInfo MipsAssembler::Branch::branch_info_[] = {
2154 // R2 short branches.
2155 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kUncondBranch
2156 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kCondBranch
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002157 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kCall
2158 // R2 near literal.
2159 { 1, 0, 0, MipsAssembler::Branch::kOffset16, 0 }, // kLiteral
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002160 // R2 long branches.
2161 { 9, 3, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongUncondBranch
2162 { 10, 4, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCondBranch
2163 { 6, 1, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCall
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002164 // R2 far literal.
2165 { 3, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kFarLiteral
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002166 // R6 short branches.
2167 { 1, 0, 1, MipsAssembler::Branch::kOffset28, 2 }, // kR6UncondBranch
2168 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kR6CondBranch
2169 // Exception: kOffset23 for beqzc/bnezc.
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002170 { 1, 0, 1, MipsAssembler::Branch::kOffset28, 2 }, // kR6Call
2171 // R6 near literal.
2172 { 1, 0, 0, MipsAssembler::Branch::kOffset21, 2 }, // kR6Literal
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002173 // R6 long branches.
2174 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongUncondBranch
2175 { 3, 1, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCondBranch
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002176 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCall
2177 // R6 far literal.
2178 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6FarLiteral
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002179};
2180
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002181// Note: make sure branch_info_[] and EmitBranch() are kept synchronized.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002182void MipsAssembler::EmitBranch(MipsAssembler::Branch* branch) {
2183 CHECK_EQ(overwriting_, true);
2184 overwrite_location_ = branch->GetLocation();
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002185 uint32_t offset = branch->GetOffset(GetBranchOrPcRelBaseForEncoding(branch));
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002186 BranchCondition condition = branch->GetCondition();
2187 Register lhs = branch->GetLeftRegister();
2188 Register rhs = branch->GetRightRegister();
2189 switch (branch->GetType()) {
2190 // R2 short branches.
2191 case Branch::kUncondBranch:
2192 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2193 B(offset);
2194 Nop(); // TODO: improve by filling the delay slot.
2195 break;
2196 case Branch::kCondBranch:
2197 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08002198 EmitBcondR2(condition, lhs, rhs, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002199 Nop(); // TODO: improve by filling the delay slot.
2200 break;
2201 case Branch::kCall:
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002202 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002203 Bal(offset);
2204 Nop(); // TODO: improve by filling the delay slot.
2205 break;
2206
2207 // R2 near literal.
2208 case Branch::kLiteral:
2209 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2210 Lw(lhs, rhs, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002211 break;
2212
2213 // R2 long branches.
2214 case Branch::kLongUncondBranch:
2215 // To get the value of the PC register we need to use the NAL instruction.
2216 // NAL clobbers the RA register. However, RA must be preserved if the
2217 // method is compiled without the entry/exit sequences that would take care
2218 // of preserving RA (typically, leaf methods don't preserve RA explicitly).
2219 // So, we need to preserve RA in some temporary storage ourselves. The AT
2220 // register can't be used for this because we need it to load a constant
2221 // which will be added to the value that NAL stores in RA. And we can't
2222 // use T9 for this in the context of the JNI compiler, which uses it
2223 // as a scratch register (see InterproceduralScratchRegister()).
2224 // If we were to add a 32-bit constant to RA using two ADDIU instructions,
2225 // we'd also need to use the ROTR instruction, which requires no less than
2226 // MIPSR2.
2227 // Perhaps, we could use T8 or one of R2's multiplier/divider registers
2228 // (LO or HI) or even a floating-point register, but that doesn't seem
2229 // like a nice solution. We may want this to work on both R6 and pre-R6.
2230 // For now simply use the stack for RA. This should be OK since for the
2231 // vast majority of code a short PC-relative branch is sufficient.
2232 // TODO: can this be improved?
2233 Push(RA);
2234 Nal();
2235 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2236 Lui(AT, High16Bits(offset));
2237 Ori(AT, AT, Low16Bits(offset));
2238 Addu(AT, AT, RA);
2239 Lw(RA, SP, 0);
2240 Jr(AT);
2241 DecreaseFrameSize(kMipsWordSize);
2242 break;
2243 case Branch::kLongCondBranch:
2244 // The comment on case 'Branch::kLongUncondBranch' applies here as well.
2245 // Note: the opposite condition branch encodes 8 as the distance, which is equal to the
2246 // number of instructions skipped:
2247 // (PUSH(IncreaseFrameSize(ADDIU) + SW) + NAL + LUI + ORI + ADDU + LW + JR).
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08002248 EmitBcondR2(Branch::OppositeCondition(condition), lhs, rhs, 8);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002249 Push(RA);
2250 Nal();
2251 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2252 Lui(AT, High16Bits(offset));
2253 Ori(AT, AT, Low16Bits(offset));
2254 Addu(AT, AT, RA);
2255 Lw(RA, SP, 0);
2256 Jr(AT);
2257 DecreaseFrameSize(kMipsWordSize);
2258 break;
2259 case Branch::kLongCall:
2260 Nal();
2261 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2262 Lui(AT, High16Bits(offset));
2263 Ori(AT, AT, Low16Bits(offset));
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002264 Addu(AT, AT, RA);
2265 Jalr(AT);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002266 Nop();
2267 break;
2268
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002269 // R2 far literal.
2270 case Branch::kFarLiteral:
2271 offset += (offset & 0x8000) << 1; // Account for sign extension in lw.
2272 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2273 Lui(AT, High16Bits(offset));
2274 Addu(AT, AT, rhs);
2275 Lw(lhs, AT, Low16Bits(offset));
2276 break;
2277
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002278 // R6 short branches.
2279 case Branch::kR6UncondBranch:
2280 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2281 Bc(offset);
2282 break;
2283 case Branch::kR6CondBranch:
2284 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08002285 EmitBcondR6(condition, lhs, rhs, offset);
2286 Nop(); // TODO: improve by filling the forbidden/delay slot.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002287 break;
2288 case Branch::kR6Call:
2289 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002290 Balc(offset);
2291 break;
2292
2293 // R6 near literal.
2294 case Branch::kR6Literal:
2295 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2296 Lwpc(lhs, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002297 break;
2298
2299 // R6 long branches.
2300 case Branch::kR6LongUncondBranch:
2301 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
2302 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2303 Auipc(AT, High16Bits(offset));
2304 Jic(AT, Low16Bits(offset));
2305 break;
2306 case Branch::kR6LongCondBranch:
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08002307 EmitBcondR6(Branch::OppositeCondition(condition), lhs, rhs, 2);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002308 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
2309 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2310 Auipc(AT, High16Bits(offset));
2311 Jic(AT, Low16Bits(offset));
2312 break;
2313 case Branch::kR6LongCall:
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002314 offset += (offset & 0x8000) << 1; // Account for sign extension in jialc.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002315 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002316 Auipc(AT, High16Bits(offset));
2317 Jialc(AT, Low16Bits(offset));
2318 break;
2319
2320 // R6 far literal.
2321 case Branch::kR6FarLiteral:
2322 offset += (offset & 0x8000) << 1; // Account for sign extension in lw.
2323 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
2324 Auipc(AT, High16Bits(offset));
2325 Lw(lhs, AT, Low16Bits(offset));
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002326 break;
2327 }
2328 CHECK_EQ(overwrite_location_, branch->GetEndLocation());
2329 CHECK_LT(branch->GetSize(), static_cast<uint32_t>(Branch::kMaxBranchSize));
2330}
2331
2332void MipsAssembler::B(MipsLabel* label) {
2333 Buncond(label);
2334}
2335
Alexey Frunzee3fb2452016-05-10 16:08:05 -07002336void MipsAssembler::Bal(MipsLabel* label) {
2337 Call(label);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002338}
2339
2340void MipsAssembler::Beq(Register rs, Register rt, MipsLabel* label) {
2341 Bcond(label, kCondEQ, rs, rt);
2342}
2343
2344void MipsAssembler::Bne(Register rs, Register rt, MipsLabel* label) {
2345 Bcond(label, kCondNE, rs, rt);
2346}
2347
2348void MipsAssembler::Beqz(Register rt, MipsLabel* label) {
2349 Bcond(label, kCondEQZ, rt);
2350}
2351
2352void MipsAssembler::Bnez(Register rt, MipsLabel* label) {
2353 Bcond(label, kCondNEZ, rt);
2354}
2355
2356void MipsAssembler::Bltz(Register rt, MipsLabel* label) {
2357 Bcond(label, kCondLTZ, rt);
2358}
2359
2360void MipsAssembler::Bgez(Register rt, MipsLabel* label) {
2361 Bcond(label, kCondGEZ, rt);
2362}
2363
2364void MipsAssembler::Blez(Register rt, MipsLabel* label) {
2365 Bcond(label, kCondLEZ, rt);
2366}
2367
2368void MipsAssembler::Bgtz(Register rt, MipsLabel* label) {
2369 Bcond(label, kCondGTZ, rt);
2370}
2371
2372void MipsAssembler::Blt(Register rs, Register rt, MipsLabel* label) {
2373 if (IsR6()) {
2374 Bcond(label, kCondLT, rs, rt);
2375 } else if (!Branch::IsNop(kCondLT, rs, rt)) {
2376 // Synthesize the instruction (not available on R2).
2377 Slt(AT, rs, rt);
2378 Bnez(AT, label);
2379 }
2380}
2381
2382void MipsAssembler::Bge(Register rs, Register rt, MipsLabel* label) {
2383 if (IsR6()) {
2384 Bcond(label, kCondGE, rs, rt);
2385 } else if (Branch::IsUncond(kCondGE, rs, rt)) {
2386 B(label);
2387 } else {
2388 // Synthesize the instruction (not available on R2).
2389 Slt(AT, rs, rt);
2390 Beqz(AT, label);
2391 }
2392}
2393
2394void MipsAssembler::Bltu(Register rs, Register rt, MipsLabel* label) {
2395 if (IsR6()) {
2396 Bcond(label, kCondLTU, rs, rt);
2397 } else if (!Branch::IsNop(kCondLTU, rs, rt)) {
2398 // Synthesize the instruction (not available on R2).
2399 Sltu(AT, rs, rt);
2400 Bnez(AT, label);
2401 }
2402}
2403
2404void MipsAssembler::Bgeu(Register rs, Register rt, MipsLabel* label) {
2405 if (IsR6()) {
2406 Bcond(label, kCondGEU, rs, rt);
2407 } else if (Branch::IsUncond(kCondGEU, rs, rt)) {
2408 B(label);
2409 } else {
2410 // Synthesize the instruction (not available on R2).
2411 Sltu(AT, rs, rt);
2412 Beqz(AT, label);
jeffhao7fbee072012-08-24 17:56:54 -07002413 }
2414}
2415
Chris Larsenb74353a2015-11-20 09:07:09 -08002416void MipsAssembler::Bc1f(MipsLabel* label) {
2417 Bc1f(0, label);
2418}
2419
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08002420void MipsAssembler::Bc1f(int cc, MipsLabel* label) {
2421 CHECK(IsUint<3>(cc)) << cc;
2422 Bcond(label, kCondF, static_cast<Register>(cc), ZERO);
2423}
2424
Chris Larsenb74353a2015-11-20 09:07:09 -08002425void MipsAssembler::Bc1t(MipsLabel* label) {
2426 Bc1t(0, label);
2427}
2428
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08002429void MipsAssembler::Bc1t(int cc, MipsLabel* label) {
2430 CHECK(IsUint<3>(cc)) << cc;
2431 Bcond(label, kCondT, static_cast<Register>(cc), ZERO);
2432}
2433
2434void MipsAssembler::Bc1eqz(FRegister ft, MipsLabel* label) {
2435 Bcond(label, kCondF, static_cast<Register>(ft), ZERO);
2436}
2437
2438void MipsAssembler::Bc1nez(FRegister ft, MipsLabel* label) {
2439 Bcond(label, kCondT, static_cast<Register>(ft), ZERO);
2440}
2441
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07002442void MipsAssembler::AdjustBaseAndOffset(Register& base,
2443 int32_t& offset,
2444 bool is_doubleword,
2445 bool is_float) {
2446 // This method is used to adjust the base register and offset pair
2447 // for a load/store when the offset doesn't fit into int16_t.
2448 // It is assumed that `base + offset` is sufficiently aligned for memory
2449 // operands that are machine word in size or smaller. For doubleword-sized
2450 // operands it's assumed that `base` is a multiple of 8, while `offset`
2451 // may be a multiple of 4 (e.g. 4-byte-aligned long and double arguments
2452 // and spilled variables on the stack accessed relative to the stack
2453 // pointer register).
2454 // We preserve the "alignment" of `offset` by adjusting it by a multiple of 8.
2455 CHECK_NE(base, AT); // Must not overwrite the register `base` while loading `offset`.
2456
2457 bool doubleword_aligned = IsAligned<kMipsDoublewordSize>(offset);
2458 bool two_accesses = is_doubleword && (!is_float || !doubleword_aligned);
2459
2460 // IsInt<16> must be passed a signed value, hence the static cast below.
2461 if (IsInt<16>(offset) &&
2462 (!two_accesses || IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2463 // Nothing to do: `offset` (and, if needed, `offset + 4`) fits into int16_t.
2464 return;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002465 }
2466
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07002467 // Remember the "(mis)alignment" of `offset`, it will be checked at the end.
2468 uint32_t misalignment = offset & (kMipsDoublewordSize - 1);
2469
2470 // Do not load the whole 32-bit `offset` if it can be represented as
2471 // a sum of two 16-bit signed offsets. This can save an instruction or two.
2472 // To simplify matters, only do this for a symmetric range of offsets from
2473 // about -64KB to about +64KB, allowing further addition of 4 when accessing
2474 // 64-bit variables with two 32-bit accesses.
2475 constexpr int32_t kMinOffsetForSimpleAdjustment = 0x7ff8; // Max int16_t that's a multiple of 8.
2476 constexpr int32_t kMaxOffsetForSimpleAdjustment = 2 * kMinOffsetForSimpleAdjustment;
2477 if (0 <= offset && offset <= kMaxOffsetForSimpleAdjustment) {
2478 Addiu(AT, base, kMinOffsetForSimpleAdjustment);
2479 offset -= kMinOffsetForSimpleAdjustment;
2480 } else if (-kMaxOffsetForSimpleAdjustment <= offset && offset < 0) {
2481 Addiu(AT, base, -kMinOffsetForSimpleAdjustment);
2482 offset += kMinOffsetForSimpleAdjustment;
2483 } else if (IsR6()) {
2484 // On R6 take advantage of the aui instruction, e.g.:
2485 // aui AT, base, offset_high
2486 // lw reg_lo, offset_low(AT)
2487 // lw reg_hi, (offset_low+4)(AT)
2488 // or when offset_low+4 overflows int16_t:
2489 // aui AT, base, offset_high
2490 // addiu AT, AT, 8
2491 // lw reg_lo, (offset_low-8)(AT)
2492 // lw reg_hi, (offset_low-4)(AT)
2493 int16_t offset_high = High16Bits(offset);
2494 int16_t offset_low = Low16Bits(offset);
2495 offset_high += (offset_low < 0) ? 1 : 0; // Account for offset sign extension in load/store.
2496 Aui(AT, base, offset_high);
2497 if (two_accesses && !IsInt<16>(static_cast<int32_t>(offset_low + kMipsWordSize))) {
2498 // Avoid overflow in the 16-bit offset of the load/store instruction when adding 4.
2499 Addiu(AT, AT, kMipsDoublewordSize);
2500 offset_low -= kMipsDoublewordSize;
2501 }
2502 offset = offset_low;
2503 } else {
2504 // Do not load the whole 32-bit `offset` if it can be represented as
2505 // a sum of three 16-bit signed offsets. This can save an instruction.
2506 // To simplify matters, only do this for a symmetric range of offsets from
2507 // about -96KB to about +96KB, allowing further addition of 4 when accessing
2508 // 64-bit variables with two 32-bit accesses.
2509 constexpr int32_t kMinOffsetForMediumAdjustment = 2 * kMinOffsetForSimpleAdjustment;
2510 constexpr int32_t kMaxOffsetForMediumAdjustment = 3 * kMinOffsetForSimpleAdjustment;
2511 if (0 <= offset && offset <= kMaxOffsetForMediumAdjustment) {
2512 Addiu(AT, base, kMinOffsetForMediumAdjustment / 2);
2513 Addiu(AT, AT, kMinOffsetForMediumAdjustment / 2);
2514 offset -= kMinOffsetForMediumAdjustment;
2515 } else if (-kMaxOffsetForMediumAdjustment <= offset && offset < 0) {
2516 Addiu(AT, base, -kMinOffsetForMediumAdjustment / 2);
2517 Addiu(AT, AT, -kMinOffsetForMediumAdjustment / 2);
2518 offset += kMinOffsetForMediumAdjustment;
2519 } else {
2520 // Now that all shorter options have been exhausted, load the full 32-bit offset.
2521 int32_t loaded_offset = RoundDown(offset, kMipsDoublewordSize);
2522 LoadConst32(AT, loaded_offset);
2523 Addu(AT, AT, base);
2524 offset -= loaded_offset;
2525 }
2526 }
2527 base = AT;
2528
2529 CHECK(IsInt<16>(offset));
2530 if (two_accesses) {
2531 CHECK(IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)));
2532 }
2533 CHECK_EQ(misalignment, offset & (kMipsDoublewordSize - 1));
2534}
2535
2536void MipsAssembler::LoadFromOffset(LoadOperandType type, Register reg, Register base,
2537 int32_t offset) {
2538 AdjustBaseAndOffset(base, offset, /* is_doubleword */ (type == kLoadDoubleword));
jeffhao7fbee072012-08-24 17:56:54 -07002539 switch (type) {
2540 case kLoadSignedByte:
2541 Lb(reg, base, offset);
2542 break;
2543 case kLoadUnsignedByte:
2544 Lbu(reg, base, offset);
2545 break;
2546 case kLoadSignedHalfword:
2547 Lh(reg, base, offset);
2548 break;
2549 case kLoadUnsignedHalfword:
2550 Lhu(reg, base, offset);
2551 break;
2552 case kLoadWord:
2553 Lw(reg, base, offset);
2554 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002555 case kLoadDoubleword:
2556 if (reg == base) {
2557 // This will clobber the base when loading the lower register. Since we have to load the
2558 // higher register as well, this will fail. Solution: reverse the order.
2559 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
2560 Lw(reg, base, offset);
2561 } else {
2562 Lw(reg, base, offset);
2563 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
2564 }
jeffhao7fbee072012-08-24 17:56:54 -07002565 break;
2566 default:
2567 LOG(FATAL) << "UNREACHABLE";
2568 }
2569}
2570
2571void MipsAssembler::LoadSFromOffset(FRegister reg, Register base, int32_t offset) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07002572 AdjustBaseAndOffset(base, offset, /* is_doubleword */ false, /* is_float */ true);
jeffhao7fbee072012-08-24 17:56:54 -07002573 Lwc1(reg, base, offset);
2574}
2575
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002576void MipsAssembler::LoadDFromOffset(FRegister reg, Register base, int32_t offset) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07002577 AdjustBaseAndOffset(base, offset, /* is_doubleword */ true, /* is_float */ true);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002578 if (offset & 0x7) {
2579 if (Is32BitFPU()) {
2580 Lwc1(reg, base, offset);
2581 Lwc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
2582 } else {
2583 // 64-bit FPU.
2584 Lwc1(reg, base, offset);
2585 Lw(T8, base, offset + kMipsWordSize);
2586 Mthc1(T8, reg);
2587 }
2588 } else {
2589 Ldc1(reg, base, offset);
2590 }
2591}
2592
2593void MipsAssembler::EmitLoad(ManagedRegister m_dst, Register src_register, int32_t src_offset,
2594 size_t size) {
2595 MipsManagedRegister dst = m_dst.AsMips();
2596 if (dst.IsNoRegister()) {
2597 CHECK_EQ(0u, size) << dst;
2598 } else if (dst.IsCoreRegister()) {
2599 CHECK_EQ(kMipsWordSize, size) << dst;
2600 LoadFromOffset(kLoadWord, dst.AsCoreRegister(), src_register, src_offset);
2601 } else if (dst.IsRegisterPair()) {
2602 CHECK_EQ(kMipsDoublewordSize, size) << dst;
2603 LoadFromOffset(kLoadDoubleword, dst.AsRegisterPairLow(), src_register, src_offset);
2604 } else if (dst.IsFRegister()) {
2605 if (size == kMipsWordSize) {
2606 LoadSFromOffset(dst.AsFRegister(), src_register, src_offset);
2607 } else {
2608 CHECK_EQ(kMipsDoublewordSize, size) << dst;
2609 LoadDFromOffset(dst.AsFRegister(), src_register, src_offset);
2610 }
2611 }
jeffhao7fbee072012-08-24 17:56:54 -07002612}
2613
2614void MipsAssembler::StoreToOffset(StoreOperandType type, Register reg, Register base,
2615 int32_t offset) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07002616 // Must not use AT as `reg`, so as not to overwrite the value being stored
2617 // with the adjusted `base`.
2618 CHECK_NE(reg, AT);
2619 AdjustBaseAndOffset(base, offset, /* is_doubleword */ (type == kStoreDoubleword));
jeffhao7fbee072012-08-24 17:56:54 -07002620 switch (type) {
2621 case kStoreByte:
2622 Sb(reg, base, offset);
2623 break;
2624 case kStoreHalfword:
2625 Sh(reg, base, offset);
2626 break;
2627 case kStoreWord:
2628 Sw(reg, base, offset);
2629 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002630 case kStoreDoubleword:
2631 CHECK_NE(reg, base);
2632 CHECK_NE(static_cast<Register>(reg + 1), base);
2633 Sw(reg, base, offset);
2634 Sw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002635 break;
2636 default:
2637 LOG(FATAL) << "UNREACHABLE";
2638 }
2639}
2640
Goran Jakovljevicff734982015-08-24 12:58:55 +00002641void MipsAssembler::StoreSToOffset(FRegister reg, Register base, int32_t offset) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07002642 AdjustBaseAndOffset(base, offset, /* is_doubleword */ false, /* is_float */ true);
jeffhao7fbee072012-08-24 17:56:54 -07002643 Swc1(reg, base, offset);
2644}
2645
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002646void MipsAssembler::StoreDToOffset(FRegister reg, Register base, int32_t offset) {
Alexey Frunzecad3a4c2016-06-07 23:40:37 -07002647 AdjustBaseAndOffset(base, offset, /* is_doubleword */ true, /* is_float */ true);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002648 if (offset & 0x7) {
2649 if (Is32BitFPU()) {
2650 Swc1(reg, base, offset);
2651 Swc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
2652 } else {
2653 // 64-bit FPU.
2654 Mfhc1(T8, reg);
2655 Swc1(reg, base, offset);
2656 Sw(T8, base, offset + kMipsWordSize);
2657 }
2658 } else {
2659 Sdc1(reg, base, offset);
2660 }
jeffhao7fbee072012-08-24 17:56:54 -07002661}
2662
David Srbeckydd973932015-04-07 20:29:48 +01002663static dwarf::Reg DWARFReg(Register reg) {
2664 return dwarf::Reg::MipsCore(static_cast<int>(reg));
2665}
2666
Ian Rogers790a6b72014-04-01 10:36:00 -07002667constexpr size_t kFramePointerSize = 4;
2668
Vladimir Marko32248382016-05-19 10:37:24 +01002669void MipsAssembler::BuildFrame(size_t frame_size,
2670 ManagedRegister method_reg,
2671 ArrayRef<const ManagedRegister> callee_save_regs,
Dmitry Petrochenkofca82202014-03-21 11:21:37 +07002672 const ManagedRegisterEntrySpills& entry_spills) {
jeffhao7fbee072012-08-24 17:56:54 -07002673 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002674 DCHECK(!overwriting_);
jeffhao7fbee072012-08-24 17:56:54 -07002675
2676 // Increase frame to required size.
2677 IncreaseFrameSize(frame_size);
2678
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002679 // Push callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07002680 int stack_offset = frame_size - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002681 StoreToOffset(kStoreWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002682 cfi_.RelOffset(DWARFReg(RA), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07002683 for (int i = callee_save_regs.size() - 1; i >= 0; --i) {
Ian Rogers790a6b72014-04-01 10:36:00 -07002684 stack_offset -= kFramePointerSize;
Vladimir Marko32248382016-05-19 10:37:24 +01002685 Register reg = callee_save_regs[i].AsMips().AsCoreRegister();
jeffhao7fbee072012-08-24 17:56:54 -07002686 StoreToOffset(kStoreWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002687 cfi_.RelOffset(DWARFReg(reg), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07002688 }
2689
2690 // Write out Method*.
2691 StoreToOffset(kStoreWord, method_reg.AsMips().AsCoreRegister(), SP, 0);
2692
2693 // Write out entry spills.
Goran Jakovljevicff734982015-08-24 12:58:55 +00002694 int32_t offset = frame_size + kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002695 for (size_t i = 0; i < entry_spills.size(); ++i) {
Goran Jakovljevicff734982015-08-24 12:58:55 +00002696 MipsManagedRegister reg = entry_spills.at(i).AsMips();
2697 if (reg.IsNoRegister()) {
2698 ManagedRegisterSpill spill = entry_spills.at(i);
2699 offset += spill.getSize();
2700 } else if (reg.IsCoreRegister()) {
2701 StoreToOffset(kStoreWord, reg.AsCoreRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002702 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002703 } else if (reg.IsFRegister()) {
2704 StoreSToOffset(reg.AsFRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002705 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002706 } else if (reg.IsDRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002707 StoreDToOffset(reg.AsOverlappingDRegisterLow(), SP, offset);
2708 offset += kMipsDoublewordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002709 }
jeffhao7fbee072012-08-24 17:56:54 -07002710 }
2711}
2712
2713void MipsAssembler::RemoveFrame(size_t frame_size,
Vladimir Marko32248382016-05-19 10:37:24 +01002714 ArrayRef<const ManagedRegister> callee_save_regs) {
jeffhao7fbee072012-08-24 17:56:54 -07002715 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002716 DCHECK(!overwriting_);
David Srbeckydd973932015-04-07 20:29:48 +01002717 cfi_.RememberState();
jeffhao7fbee072012-08-24 17:56:54 -07002718
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002719 // Pop callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07002720 int stack_offset = frame_size - (callee_save_regs.size() * kFramePointerSize) - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002721 for (size_t i = 0; i < callee_save_regs.size(); ++i) {
Vladimir Marko32248382016-05-19 10:37:24 +01002722 Register reg = callee_save_regs[i].AsMips().AsCoreRegister();
jeffhao7fbee072012-08-24 17:56:54 -07002723 LoadFromOffset(kLoadWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002724 cfi_.Restore(DWARFReg(reg));
Ian Rogers790a6b72014-04-01 10:36:00 -07002725 stack_offset += kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002726 }
2727 LoadFromOffset(kLoadWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002728 cfi_.Restore(DWARFReg(RA));
jeffhao7fbee072012-08-24 17:56:54 -07002729
2730 // Decrease frame to required size.
2731 DecreaseFrameSize(frame_size);
jeffhao07030602012-09-26 14:33:14 -07002732
2733 // Then jump to the return address.
2734 Jr(RA);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002735 Nop();
David Srbeckydd973932015-04-07 20:29:48 +01002736
2737 // The CFI should be restored for any code that follows the exit block.
2738 cfi_.RestoreState();
2739 cfi_.DefCFAOffset(frame_size);
jeffhao7fbee072012-08-24 17:56:54 -07002740}
2741
2742void MipsAssembler::IncreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002743 CHECK_ALIGNED(adjust, kFramePointerSize);
2744 Addiu32(SP, SP, -adjust);
David Srbeckydd973932015-04-07 20:29:48 +01002745 cfi_.AdjustCFAOffset(adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002746 if (overwriting_) {
2747 cfi_.OverrideDelayedPC(overwrite_location_);
2748 }
jeffhao7fbee072012-08-24 17:56:54 -07002749}
2750
2751void MipsAssembler::DecreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002752 CHECK_ALIGNED(adjust, kFramePointerSize);
2753 Addiu32(SP, SP, adjust);
David Srbeckydd973932015-04-07 20:29:48 +01002754 cfi_.AdjustCFAOffset(-adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002755 if (overwriting_) {
2756 cfi_.OverrideDelayedPC(overwrite_location_);
2757 }
jeffhao7fbee072012-08-24 17:56:54 -07002758}
2759
2760void MipsAssembler::Store(FrameOffset dest, ManagedRegister msrc, size_t size) {
2761 MipsManagedRegister src = msrc.AsMips();
2762 if (src.IsNoRegister()) {
2763 CHECK_EQ(0u, size);
2764 } else if (src.IsCoreRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002765 CHECK_EQ(kMipsWordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07002766 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2767 } else if (src.IsRegisterPair()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002768 CHECK_EQ(kMipsDoublewordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07002769 StoreToOffset(kStoreWord, src.AsRegisterPairLow(), SP, dest.Int32Value());
2770 StoreToOffset(kStoreWord, src.AsRegisterPairHigh(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002771 SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002772 } else if (src.IsFRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002773 if (size == kMipsWordSize) {
2774 StoreSToOffset(src.AsFRegister(), SP, dest.Int32Value());
2775 } else {
2776 CHECK_EQ(kMipsDoublewordSize, size);
2777 StoreDToOffset(src.AsFRegister(), SP, dest.Int32Value());
2778 }
jeffhao7fbee072012-08-24 17:56:54 -07002779 }
2780}
2781
2782void MipsAssembler::StoreRef(FrameOffset dest, ManagedRegister msrc) {
2783 MipsManagedRegister src = msrc.AsMips();
2784 CHECK(src.IsCoreRegister());
2785 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2786}
2787
2788void MipsAssembler::StoreRawPtr(FrameOffset dest, ManagedRegister msrc) {
2789 MipsManagedRegister src = msrc.AsMips();
2790 CHECK(src.IsCoreRegister());
2791 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2792}
2793
2794void MipsAssembler::StoreImmediateToFrame(FrameOffset dest, uint32_t imm,
2795 ManagedRegister mscratch) {
2796 MipsManagedRegister scratch = mscratch.AsMips();
2797 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002798 LoadConst32(scratch.AsCoreRegister(), imm);
jeffhao7fbee072012-08-24 17:56:54 -07002799 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2800}
2801
Andreas Gampe542451c2016-07-26 09:02:02 -07002802void MipsAssembler::StoreImmediateToThread32(ThreadOffset32 dest,
2803 uint32_t imm,
jeffhao7fbee072012-08-24 17:56:54 -07002804 ManagedRegister mscratch) {
2805 MipsManagedRegister scratch = mscratch.AsMips();
2806 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002807 // Is this function even referenced anywhere else in the code?
2808 LoadConst32(scratch.AsCoreRegister(), imm);
2809 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), S1, dest.Int32Value());
2810}
2811
Andreas Gampe542451c2016-07-26 09:02:02 -07002812void MipsAssembler::StoreStackOffsetToThread32(ThreadOffset32 thr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002813 FrameOffset fr_offs,
2814 ManagedRegister mscratch) {
2815 MipsManagedRegister scratch = mscratch.AsMips();
2816 CHECK(scratch.IsCoreRegister()) << scratch;
2817 Addiu32(scratch.AsCoreRegister(), SP, fr_offs.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002818 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2819 S1, thr_offs.Int32Value());
2820}
2821
Andreas Gampe542451c2016-07-26 09:02:02 -07002822void MipsAssembler::StoreStackPointerToThread32(ThreadOffset32 thr_offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002823 StoreToOffset(kStoreWord, SP, S1, thr_offs.Int32Value());
2824}
2825
2826void MipsAssembler::StoreSpanning(FrameOffset dest, ManagedRegister msrc,
2827 FrameOffset in_off, ManagedRegister mscratch) {
2828 MipsManagedRegister src = msrc.AsMips();
2829 MipsManagedRegister scratch = mscratch.AsMips();
2830 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2831 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, in_off.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002832 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002833}
2834
2835void MipsAssembler::Load(ManagedRegister mdest, FrameOffset src, size_t size) {
2836 return EmitLoad(mdest, SP, src.Int32Value(), size);
2837}
2838
Andreas Gampe542451c2016-07-26 09:02:02 -07002839void MipsAssembler::LoadFromThread32(ManagedRegister mdest, ThreadOffset32 src, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002840 return EmitLoad(mdest, S1, src.Int32Value(), size);
2841}
2842
2843void MipsAssembler::LoadRef(ManagedRegister mdest, FrameOffset src) {
2844 MipsManagedRegister dest = mdest.AsMips();
2845 CHECK(dest.IsCoreRegister());
2846 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), SP, src.Int32Value());
2847}
2848
Mathieu Chartiere401d142015-04-22 13:56:20 -07002849void MipsAssembler::LoadRef(ManagedRegister mdest, ManagedRegister base, MemberOffset offs,
Roland Levillain4d027112015-07-01 15:41:14 +01002850 bool unpoison_reference) {
jeffhao7fbee072012-08-24 17:56:54 -07002851 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002852 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002853 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2854 base.AsMips().AsCoreRegister(), offs.Int32Value());
Roland Levillain4d027112015-07-01 15:41:14 +01002855 if (kPoisonHeapReferences && unpoison_reference) {
Hiroshi Yamauchie63a7452014-02-27 14:44:36 -08002856 Subu(dest.AsCoreRegister(), ZERO, dest.AsCoreRegister());
2857 }
jeffhao7fbee072012-08-24 17:56:54 -07002858}
2859
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002860void MipsAssembler::LoadRawPtr(ManagedRegister mdest, ManagedRegister base, Offset offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002861 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002862 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002863 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2864 base.AsMips().AsCoreRegister(), offs.Int32Value());
2865}
2866
Andreas Gampe542451c2016-07-26 09:02:02 -07002867void MipsAssembler::LoadRawPtrFromThread32(ManagedRegister mdest, ThreadOffset32 offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002868 MipsManagedRegister dest = mdest.AsMips();
2869 CHECK(dest.IsCoreRegister());
2870 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), S1, offs.Int32Value());
2871}
2872
2873void MipsAssembler::SignExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2874 UNIMPLEMENTED(FATAL) << "no sign extension necessary for mips";
2875}
2876
2877void MipsAssembler::ZeroExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2878 UNIMPLEMENTED(FATAL) << "no zero extension necessary for mips";
2879}
2880
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002881void MipsAssembler::Move(ManagedRegister mdest, ManagedRegister msrc, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002882 MipsManagedRegister dest = mdest.AsMips();
2883 MipsManagedRegister src = msrc.AsMips();
2884 if (!dest.Equals(src)) {
2885 if (dest.IsCoreRegister()) {
2886 CHECK(src.IsCoreRegister()) << src;
2887 Move(dest.AsCoreRegister(), src.AsCoreRegister());
2888 } else if (dest.IsFRegister()) {
2889 CHECK(src.IsFRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002890 if (size == kMipsWordSize) {
2891 MovS(dest.AsFRegister(), src.AsFRegister());
2892 } else {
2893 CHECK_EQ(kMipsDoublewordSize, size);
2894 MovD(dest.AsFRegister(), src.AsFRegister());
2895 }
jeffhao7fbee072012-08-24 17:56:54 -07002896 } else if (dest.IsDRegister()) {
2897 CHECK(src.IsDRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002898 MovD(dest.AsOverlappingDRegisterLow(), src.AsOverlappingDRegisterLow());
jeffhao7fbee072012-08-24 17:56:54 -07002899 } else {
2900 CHECK(dest.IsRegisterPair()) << dest;
2901 CHECK(src.IsRegisterPair()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002902 // Ensure that the first move doesn't clobber the input of the second.
jeffhao7fbee072012-08-24 17:56:54 -07002903 if (src.AsRegisterPairHigh() != dest.AsRegisterPairLow()) {
2904 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2905 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2906 } else {
2907 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2908 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2909 }
2910 }
2911 }
2912}
2913
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002914void MipsAssembler::CopyRef(FrameOffset dest, FrameOffset src, ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002915 MipsManagedRegister scratch = mscratch.AsMips();
2916 CHECK(scratch.IsCoreRegister()) << scratch;
2917 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2918 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2919}
2920
Ian Rogersdd7624d2014-03-14 17:43:00 -07002921void MipsAssembler::CopyRawPtrFromThread32(FrameOffset fr_offs,
Andreas Gampe542451c2016-07-26 09:02:02 -07002922 ThreadOffset32 thr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002923 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002924 MipsManagedRegister scratch = mscratch.AsMips();
2925 CHECK(scratch.IsCoreRegister()) << scratch;
2926 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2927 S1, thr_offs.Int32Value());
2928 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2929 SP, fr_offs.Int32Value());
2930}
2931
Andreas Gampe542451c2016-07-26 09:02:02 -07002932void MipsAssembler::CopyRawPtrToThread32(ThreadOffset32 thr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002933 FrameOffset fr_offs,
2934 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002935 MipsManagedRegister scratch = mscratch.AsMips();
2936 CHECK(scratch.IsCoreRegister()) << scratch;
2937 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2938 SP, fr_offs.Int32Value());
2939 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2940 S1, thr_offs.Int32Value());
2941}
2942
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002943void MipsAssembler::Copy(FrameOffset dest, FrameOffset src, ManagedRegister mscratch, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002944 MipsManagedRegister scratch = mscratch.AsMips();
2945 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002946 CHECK(size == kMipsWordSize || size == kMipsDoublewordSize) << size;
2947 if (size == kMipsWordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002948 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2949 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002950 } else if (size == kMipsDoublewordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002951 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2952 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002953 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value() + kMipsWordSize);
2954 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002955 }
2956}
2957
2958void MipsAssembler::Copy(FrameOffset dest, ManagedRegister src_base, Offset src_offset,
2959 ManagedRegister mscratch, size_t size) {
2960 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002961 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002962 LoadFromOffset(kLoadWord, scratch, src_base.AsMips().AsCoreRegister(), src_offset.Int32Value());
2963 StoreToOffset(kStoreWord, scratch, SP, dest.Int32Value());
2964}
2965
2966void MipsAssembler::Copy(ManagedRegister dest_base, Offset dest_offset, FrameOffset src,
2967 ManagedRegister mscratch, size_t size) {
2968 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002969 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002970 LoadFromOffset(kLoadWord, scratch, SP, src.Int32Value());
2971 StoreToOffset(kStoreWord, scratch, dest_base.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2972}
2973
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002974void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2975 FrameOffset src_base ATTRIBUTE_UNUSED,
2976 Offset src_offset ATTRIBUTE_UNUSED,
2977 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2978 size_t size ATTRIBUTE_UNUSED) {
2979 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002980}
2981
2982void MipsAssembler::Copy(ManagedRegister dest, Offset dest_offset,
2983 ManagedRegister src, Offset src_offset,
2984 ManagedRegister mscratch, size_t size) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002985 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002986 Register scratch = mscratch.AsMips().AsCoreRegister();
2987 LoadFromOffset(kLoadWord, scratch, src.AsMips().AsCoreRegister(), src_offset.Int32Value());
2988 StoreToOffset(kStoreWord, scratch, dest.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2989}
2990
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002991void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2992 Offset dest_offset ATTRIBUTE_UNUSED,
2993 FrameOffset src ATTRIBUTE_UNUSED,
2994 Offset src_offset ATTRIBUTE_UNUSED,
2995 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2996 size_t size ATTRIBUTE_UNUSED) {
2997 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002998}
2999
3000void MipsAssembler::MemoryBarrier(ManagedRegister) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003001 // TODO: sync?
3002 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07003003}
3004
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07003005void MipsAssembler::CreateHandleScopeEntry(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003006 FrameOffset handle_scope_offset,
3007 ManagedRegister min_reg,
3008 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07003009 MipsManagedRegister out_reg = mout_reg.AsMips();
3010 MipsManagedRegister in_reg = min_reg.AsMips();
3011 CHECK(in_reg.IsNoRegister() || in_reg.IsCoreRegister()) << in_reg;
3012 CHECK(out_reg.IsCoreRegister()) << out_reg;
3013 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003014 MipsLabel null_arg;
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07003015 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
3016 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003017 // E.g. out_reg = (handle == 0) ? 0 : (SP+handle_offset).
jeffhao7fbee072012-08-24 17:56:54 -07003018 if (in_reg.IsNoRegister()) {
3019 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07003020 SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07003021 in_reg = out_reg;
3022 }
3023 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003024 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07003025 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003026 Beqz(in_reg.AsCoreRegister(), &null_arg);
3027 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
3028 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07003029 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003030 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07003031 }
3032}
3033
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07003034void MipsAssembler::CreateHandleScopeEntry(FrameOffset out_off,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003035 FrameOffset handle_scope_offset,
3036 ManagedRegister mscratch,
3037 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07003038 MipsManagedRegister scratch = mscratch.AsMips();
3039 CHECK(scratch.IsCoreRegister()) << scratch;
3040 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003041 MipsLabel null_arg;
3042 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07003043 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
3044 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003045 // E.g. scratch = (scratch == 0) ? 0 : (SP+handle_scope_offset).
3046 Beqz(scratch.AsCoreRegister(), &null_arg);
3047 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
3048 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07003049 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003050 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07003051 }
3052 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, out_off.Int32Value());
3053}
3054
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07003055// Given a handle scope entry, load the associated reference.
3056void MipsAssembler::LoadReferenceFromHandleScope(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003057 ManagedRegister min_reg) {
jeffhao7fbee072012-08-24 17:56:54 -07003058 MipsManagedRegister out_reg = mout_reg.AsMips();
3059 MipsManagedRegister in_reg = min_reg.AsMips();
3060 CHECK(out_reg.IsCoreRegister()) << out_reg;
3061 CHECK(in_reg.IsCoreRegister()) << in_reg;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003062 MipsLabel null_arg;
jeffhao7fbee072012-08-24 17:56:54 -07003063 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003064 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07003065 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003066 Beqz(in_reg.AsCoreRegister(), &null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07003067 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
3068 in_reg.AsCoreRegister(), 0);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003069 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07003070}
3071
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003072void MipsAssembler::VerifyObject(ManagedRegister src ATTRIBUTE_UNUSED,
3073 bool could_be_null ATTRIBUTE_UNUSED) {
3074 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07003075}
3076
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003077void MipsAssembler::VerifyObject(FrameOffset src ATTRIBUTE_UNUSED,
3078 bool could_be_null ATTRIBUTE_UNUSED) {
3079 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07003080}
3081
3082void MipsAssembler::Call(ManagedRegister mbase, Offset offset, ManagedRegister mscratch) {
3083 MipsManagedRegister base = mbase.AsMips();
3084 MipsManagedRegister scratch = mscratch.AsMips();
3085 CHECK(base.IsCoreRegister()) << base;
3086 CHECK(scratch.IsCoreRegister()) << scratch;
3087 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
3088 base.AsCoreRegister(), offset.Int32Value());
3089 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003090 Nop();
3091 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07003092}
3093
3094void MipsAssembler::Call(FrameOffset base, Offset offset, ManagedRegister mscratch) {
3095 MipsManagedRegister scratch = mscratch.AsMips();
3096 CHECK(scratch.IsCoreRegister()) << scratch;
3097 // Call *(*(SP + base) + offset)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003098 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, base.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07003099 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
3100 scratch.AsCoreRegister(), offset.Int32Value());
3101 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003102 Nop();
3103 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07003104}
3105
Andreas Gampe542451c2016-07-26 09:02:02 -07003106void MipsAssembler::CallFromThread32(ThreadOffset32 offset ATTRIBUTE_UNUSED,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003107 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
Ian Rogers468532e2013-08-05 10:56:33 -07003108 UNIMPLEMENTED(FATAL) << "no mips implementation";
jeffhao7fbee072012-08-24 17:56:54 -07003109}
3110
3111void MipsAssembler::GetCurrentThread(ManagedRegister tr) {
3112 Move(tr.AsMips().AsCoreRegister(), S1);
3113}
3114
3115void MipsAssembler::GetCurrentThread(FrameOffset offset,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003116 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
jeffhao7fbee072012-08-24 17:56:54 -07003117 StoreToOffset(kStoreWord, S1, SP, offset.Int32Value());
3118}
3119
jeffhao7fbee072012-08-24 17:56:54 -07003120void MipsAssembler::ExceptionPoll(ManagedRegister mscratch, size_t stack_adjust) {
3121 MipsManagedRegister scratch = mscratch.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003122 exception_blocks_.emplace_back(scratch, stack_adjust);
jeffhao7fbee072012-08-24 17:56:54 -07003123 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
Andreas Gampe542451c2016-07-26 09:02:02 -07003124 S1, Thread::ExceptionOffset<kMipsPointerSize>().Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003125 // TODO: on MIPS32R6 prefer Bnezc(scratch.AsCoreRegister(), slow.Entry());
3126 // as the NAL instruction (occurring in long R2 branches) may become deprecated.
3127 // For now use common for R2 and R6 instructions as this code must execute on both.
3128 Bnez(scratch.AsCoreRegister(), exception_blocks_.back().Entry());
jeffhao7fbee072012-08-24 17:56:54 -07003129}
3130
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003131void MipsAssembler::EmitExceptionPoll(MipsExceptionSlowPath* exception) {
3132 Bind(exception->Entry());
3133 if (exception->stack_adjust_ != 0) { // Fix up the frame.
3134 DecreaseFrameSize(exception->stack_adjust_);
jeffhao7fbee072012-08-24 17:56:54 -07003135 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003136 // Pass exception object as argument.
3137 // Don't care about preserving A0 as this call won't return.
3138 CheckEntrypointTypes<kQuickDeliverException, void, mirror::Object*>();
3139 Move(A0, exception->scratch_.AsCoreRegister());
3140 // Set up call to Thread::Current()->pDeliverException.
3141 LoadFromOffset(kLoadWord, T9, S1,
Andreas Gampe542451c2016-07-26 09:02:02 -07003142 QUICK_ENTRYPOINT_OFFSET(kMipsPointerSize, pDeliverException).Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02003143 Jr(T9);
3144 Nop();
3145
3146 // Call never returns.
3147 Break();
jeffhao7fbee072012-08-24 17:56:54 -07003148}
3149
3150} // namespace mips
3151} // namespace art