blob: afca8adcbb235df6174e0e7da0faa67cb10e25e1 [file] [log] [blame]
jeffhao7fbee072012-08-24 17:56:54 -07001/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "assembler_mips.h"
18
Vladimir Marko80afd022015-05-19 18:08:00 +010019#include "base/bit_utils.h"
Elliott Hughes1aa246d2012-12-13 09:29:36 -080020#include "base/casts.h"
Ian Rogers166db042013-07-26 12:05:57 -070021#include "entrypoints/quick/quick_entrypoints.h"
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020022#include "entrypoints/quick/quick_entrypoints_enum.h"
jeffhao7fbee072012-08-24 17:56:54 -070023#include "memory_region.h"
jeffhao7fbee072012-08-24 17:56:54 -070024#include "thread.h"
25
26namespace art {
27namespace mips {
jeffhao7fbee072012-08-24 17:56:54 -070028
jeffhao7fbee072012-08-24 17:56:54 -070029std::ostream& operator<<(std::ostream& os, const DRegister& rhs) {
30 if (rhs >= D0 && rhs < kNumberOfDRegisters) {
31 os << "d" << static_cast<int>(rhs);
32 } else {
33 os << "DRegister[" << static_cast<int>(rhs) << "]";
34 }
35 return os;
36}
37
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020038void MipsAssembler::FinalizeCode() {
39 for (auto& exception_block : exception_blocks_) {
40 EmitExceptionPoll(&exception_block);
41 }
42 PromoteBranches();
43}
44
45void MipsAssembler::FinalizeInstructions(const MemoryRegion& region) {
Vladimir Marko10ef6942015-10-22 15:25:54 +010046 size_t number_of_delayed_adjust_pcs = cfi().NumberOfDelayedAdvancePCs();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +020047 EmitBranches();
48 Assembler::FinalizeInstructions(region);
Vladimir Marko10ef6942015-10-22 15:25:54 +010049 PatchCFI(number_of_delayed_adjust_pcs);
50}
51
52void MipsAssembler::PatchCFI(size_t number_of_delayed_adjust_pcs) {
53 if (cfi().NumberOfDelayedAdvancePCs() == 0u) {
54 DCHECK_EQ(number_of_delayed_adjust_pcs, 0u);
55 return;
56 }
57
58 typedef DebugFrameOpCodeWriterForAssembler::DelayedAdvancePC DelayedAdvancePC;
59 const auto data = cfi().ReleaseStreamAndPrepareForDelayedAdvancePC();
60 const std::vector<uint8_t>& old_stream = data.first;
61 const std::vector<DelayedAdvancePC>& advances = data.second;
62
63 // PCs recorded before EmitBranches() need to be adjusted.
64 // PCs recorded during EmitBranches() are already adjusted.
65 // Both ranges are separately sorted but they may overlap.
66 if (kIsDebugBuild) {
67 auto cmp = [](const DelayedAdvancePC& lhs, const DelayedAdvancePC& rhs) {
68 return lhs.pc < rhs.pc;
69 };
70 CHECK(std::is_sorted(advances.begin(), advances.begin() + number_of_delayed_adjust_pcs, cmp));
71 CHECK(std::is_sorted(advances.begin() + number_of_delayed_adjust_pcs, advances.end(), cmp));
72 }
73
74 // Append initial CFI data if any.
75 size_t size = advances.size();
76 DCHECK_NE(size, 0u);
77 cfi().AppendRawData(old_stream, 0u, advances[0].stream_pos);
78 // Emit PC adjustments interleaved with the old CFI stream.
79 size_t adjust_pos = 0u;
80 size_t late_emit_pos = number_of_delayed_adjust_pcs;
81 while (adjust_pos != number_of_delayed_adjust_pcs || late_emit_pos != size) {
82 size_t adjusted_pc = (adjust_pos != number_of_delayed_adjust_pcs)
83 ? GetAdjustedPosition(advances[adjust_pos].pc)
84 : static_cast<size_t>(-1);
85 size_t late_emit_pc = (late_emit_pos != size)
86 ? advances[late_emit_pos].pc
87 : static_cast<size_t>(-1);
88 size_t advance_pc = std::min(adjusted_pc, late_emit_pc);
89 DCHECK_NE(advance_pc, static_cast<size_t>(-1));
90 size_t entry = (adjusted_pc <= late_emit_pc) ? adjust_pos : late_emit_pos;
91 if (adjusted_pc <= late_emit_pc) {
92 ++adjust_pos;
93 } else {
94 ++late_emit_pos;
95 }
96 cfi().AdvancePC(advance_pc);
97 size_t end_pos = (entry + 1u == size) ? old_stream.size() : advances[entry + 1u].stream_pos;
98 cfi().AppendRawData(old_stream, advances[entry].stream_pos, end_pos);
99 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200100}
101
102void MipsAssembler::EmitBranches() {
103 CHECK(!overwriting_);
104 // Switch from appending instructions at the end of the buffer to overwriting
105 // existing instructions (branch placeholders) in the buffer.
106 overwriting_ = true;
107 for (auto& branch : branches_) {
108 EmitBranch(&branch);
109 }
110 overwriting_ = false;
111}
112
113void MipsAssembler::Emit(uint32_t value) {
114 if (overwriting_) {
115 // Branches to labels are emitted into their placeholders here.
116 buffer_.Store<uint32_t>(overwrite_location_, value);
117 overwrite_location_ += sizeof(uint32_t);
118 } else {
119 // Other instructions are simply appended at the end here.
120 AssemblerBuffer::EnsureCapacity ensured(&buffer_);
121 buffer_.Emit<uint32_t>(value);
122 }
jeffhao7fbee072012-08-24 17:56:54 -0700123}
124
125void MipsAssembler::EmitR(int opcode, Register rs, Register rt, Register rd, int shamt, int funct) {
126 CHECK_NE(rs, kNoRegister);
127 CHECK_NE(rt, kNoRegister);
128 CHECK_NE(rd, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200129 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
130 static_cast<uint32_t>(rs) << kRsShift |
131 static_cast<uint32_t>(rt) << kRtShift |
132 static_cast<uint32_t>(rd) << kRdShift |
133 shamt << kShamtShift |
134 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700135 Emit(encoding);
136}
137
138void MipsAssembler::EmitI(int opcode, Register rs, Register rt, uint16_t imm) {
139 CHECK_NE(rs, kNoRegister);
140 CHECK_NE(rt, kNoRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200141 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
142 static_cast<uint32_t>(rs) << kRsShift |
143 static_cast<uint32_t>(rt) << kRtShift |
144 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700145 Emit(encoding);
146}
147
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200148void MipsAssembler::EmitI21(int opcode, Register rs, uint32_t imm21) {
149 CHECK_NE(rs, kNoRegister);
150 CHECK(IsUint<21>(imm21)) << imm21;
151 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
152 static_cast<uint32_t>(rs) << kRsShift |
153 imm21;
jeffhao7fbee072012-08-24 17:56:54 -0700154 Emit(encoding);
155}
156
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200157void MipsAssembler::EmitI26(int opcode, uint32_t imm26) {
158 CHECK(IsUint<26>(imm26)) << imm26;
159 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift | imm26;
160 Emit(encoding);
161}
162
163void MipsAssembler::EmitFR(int opcode, int fmt, FRegister ft, FRegister fs, FRegister fd,
164 int funct) {
jeffhao7fbee072012-08-24 17:56:54 -0700165 CHECK_NE(ft, kNoFRegister);
166 CHECK_NE(fs, kNoFRegister);
167 CHECK_NE(fd, kNoFRegister);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200168 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
169 fmt << kFmtShift |
170 static_cast<uint32_t>(ft) << kFtShift |
171 static_cast<uint32_t>(fs) << kFsShift |
172 static_cast<uint32_t>(fd) << kFdShift |
173 funct;
jeffhao7fbee072012-08-24 17:56:54 -0700174 Emit(encoding);
175}
176
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200177void MipsAssembler::EmitFI(int opcode, int fmt, FRegister ft, uint16_t imm) {
178 CHECK_NE(ft, kNoFRegister);
179 uint32_t encoding = static_cast<uint32_t>(opcode) << kOpcodeShift |
180 fmt << kFmtShift |
181 static_cast<uint32_t>(ft) << kFtShift |
182 imm;
jeffhao7fbee072012-08-24 17:56:54 -0700183 Emit(encoding);
184}
185
jeffhao7fbee072012-08-24 17:56:54 -0700186void MipsAssembler::Addu(Register rd, Register rs, Register rt) {
187 EmitR(0, rs, rt, rd, 0, 0x21);
188}
189
jeffhao7fbee072012-08-24 17:56:54 -0700190void MipsAssembler::Addiu(Register rt, Register rs, uint16_t imm16) {
191 EmitI(0x9, rs, rt, imm16);
192}
193
jeffhao7fbee072012-08-24 17:56:54 -0700194void MipsAssembler::Subu(Register rd, Register rs, Register rt) {
195 EmitR(0, rs, rt, rd, 0, 0x23);
196}
197
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200198void MipsAssembler::MultR2(Register rs, Register rt) {
199 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700200 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x18);
201}
202
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200203void MipsAssembler::MultuR2(Register rs, Register rt) {
204 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700205 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x19);
206}
207
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200208void MipsAssembler::DivR2(Register rs, Register rt) {
209 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700210 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1a);
211}
212
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200213void MipsAssembler::DivuR2(Register rs, Register rt) {
214 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700215 EmitR(0, rs, rt, static_cast<Register>(0), 0, 0x1b);
216}
217
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200218void MipsAssembler::MulR2(Register rd, Register rs, Register rt) {
219 CHECK(!IsR6());
220 EmitR(0x1c, rs, rt, rd, 0, 2);
221}
222
223void MipsAssembler::DivR2(Register rd, Register rs, Register rt) {
224 CHECK(!IsR6());
225 DivR2(rs, rt);
226 Mflo(rd);
227}
228
229void MipsAssembler::ModR2(Register rd, Register rs, Register rt) {
230 CHECK(!IsR6());
231 DivR2(rs, rt);
232 Mfhi(rd);
233}
234
235void MipsAssembler::DivuR2(Register rd, Register rs, Register rt) {
236 CHECK(!IsR6());
237 DivuR2(rs, rt);
238 Mflo(rd);
239}
240
241void MipsAssembler::ModuR2(Register rd, Register rs, Register rt) {
242 CHECK(!IsR6());
243 DivuR2(rs, rt);
244 Mfhi(rd);
245}
246
247void MipsAssembler::MulR6(Register rd, Register rs, Register rt) {
248 CHECK(IsR6());
249 EmitR(0, rs, rt, rd, 2, 0x18);
250}
251
Alexey Frunze7e99e052015-11-24 19:28:01 -0800252void MipsAssembler::MuhR6(Register rd, Register rs, Register rt) {
253 CHECK(IsR6());
254 EmitR(0, rs, rt, rd, 3, 0x18);
255}
256
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200257void MipsAssembler::MuhuR6(Register rd, Register rs, Register rt) {
258 CHECK(IsR6());
259 EmitR(0, rs, rt, rd, 3, 0x19);
260}
261
262void MipsAssembler::DivR6(Register rd, Register rs, Register rt) {
263 CHECK(IsR6());
264 EmitR(0, rs, rt, rd, 2, 0x1a);
265}
266
267void MipsAssembler::ModR6(Register rd, Register rs, Register rt) {
268 CHECK(IsR6());
269 EmitR(0, rs, rt, rd, 3, 0x1a);
270}
271
272void MipsAssembler::DivuR6(Register rd, Register rs, Register rt) {
273 CHECK(IsR6());
274 EmitR(0, rs, rt, rd, 2, 0x1b);
275}
276
277void MipsAssembler::ModuR6(Register rd, Register rs, Register rt) {
278 CHECK(IsR6());
279 EmitR(0, rs, rt, rd, 3, 0x1b);
280}
281
jeffhao7fbee072012-08-24 17:56:54 -0700282void MipsAssembler::And(Register rd, Register rs, Register rt) {
283 EmitR(0, rs, rt, rd, 0, 0x24);
284}
285
286void MipsAssembler::Andi(Register rt, Register rs, uint16_t imm16) {
287 EmitI(0xc, rs, rt, imm16);
288}
289
290void MipsAssembler::Or(Register rd, Register rs, Register rt) {
291 EmitR(0, rs, rt, rd, 0, 0x25);
292}
293
294void MipsAssembler::Ori(Register rt, Register rs, uint16_t imm16) {
295 EmitI(0xd, rs, rt, imm16);
296}
297
298void MipsAssembler::Xor(Register rd, Register rs, Register rt) {
299 EmitR(0, rs, rt, rd, 0, 0x26);
300}
301
302void MipsAssembler::Xori(Register rt, Register rs, uint16_t imm16) {
303 EmitI(0xe, rs, rt, imm16);
304}
305
306void MipsAssembler::Nor(Register rd, Register rs, Register rt) {
307 EmitR(0, rs, rt, rd, 0, 0x27);
308}
309
Chris Larsene3845472015-11-18 12:27:15 -0800310void MipsAssembler::Movz(Register rd, Register rs, Register rt) {
311 CHECK(!IsR6());
312 EmitR(0, rs, rt, rd, 0, 0x0A);
313}
314
315void MipsAssembler::Movn(Register rd, Register rs, Register rt) {
316 CHECK(!IsR6());
317 EmitR(0, rs, rt, rd, 0, 0x0B);
318}
319
320void MipsAssembler::Seleqz(Register rd, Register rs, Register rt) {
321 CHECK(IsR6());
322 EmitR(0, rs, rt, rd, 0, 0x35);
323}
324
325void MipsAssembler::Selnez(Register rd, Register rs, Register rt) {
326 CHECK(IsR6());
327 EmitR(0, rs, rt, rd, 0, 0x37);
328}
329
330void MipsAssembler::ClzR6(Register rd, Register rs) {
331 CHECK(IsR6());
332 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x10);
333}
334
335void MipsAssembler::ClzR2(Register rd, Register rs) {
336 CHECK(!IsR6());
337 EmitR(0x1C, rs, rd, rd, 0, 0x20);
338}
339
340void MipsAssembler::CloR6(Register rd, Register rs) {
341 CHECK(IsR6());
342 EmitR(0, rs, static_cast<Register>(0), rd, 0x01, 0x11);
343}
344
345void MipsAssembler::CloR2(Register rd, Register rs) {
346 CHECK(!IsR6());
347 EmitR(0x1C, rs, rd, rd, 0, 0x21);
348}
349
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200350void MipsAssembler::Seb(Register rd, Register rt) {
351 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x10, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700352}
353
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200354void MipsAssembler::Seh(Register rd, Register rt) {
355 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x18, 0x20);
jeffhao7fbee072012-08-24 17:56:54 -0700356}
357
Chris Larsen3f8bf652015-10-28 10:08:56 -0700358void MipsAssembler::Wsbh(Register rd, Register rt) {
359 EmitR(0x1f, static_cast<Register>(0), rt, rd, 2, 0x20);
360}
361
Chris Larsen70014c82015-11-18 12:26:08 -0800362void MipsAssembler::Bitswap(Register rd, Register rt) {
363 CHECK(IsR6());
364 EmitR(0x1f, static_cast<Register>(0), rt, rd, 0x0, 0x20);
365}
366
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200367void MipsAssembler::Sll(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700368 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200369 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x00);
jeffhao7fbee072012-08-24 17:56:54 -0700370}
371
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200372void MipsAssembler::Srl(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700373 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200374 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x02);
375}
376
Chris Larsen3f8bf652015-10-28 10:08:56 -0700377void MipsAssembler::Rotr(Register rd, Register rt, int shamt) {
378 CHECK(IsUint<5>(shamt)) << shamt;
379 EmitR(0, static_cast<Register>(1), rt, rd, shamt, 0x02);
380}
381
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200382void MipsAssembler::Sra(Register rd, Register rt, int shamt) {
Chris Larsen3f8bf652015-10-28 10:08:56 -0700383 CHECK(IsUint<5>(shamt)) << shamt;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200384 EmitR(0, static_cast<Register>(0), rt, rd, shamt, 0x03);
385}
386
387void MipsAssembler::Sllv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700388 EmitR(0, rs, rt, rd, 0, 0x04);
389}
390
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200391void MipsAssembler::Srlv(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700392 EmitR(0, rs, rt, rd, 0, 0x06);
393}
394
Chris Larsene16ce5a2015-11-18 12:30:20 -0800395void MipsAssembler::Rotrv(Register rd, Register rt, Register rs) {
396 EmitR(0, rs, rt, rd, 1, 0x06);
397}
398
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200399void MipsAssembler::Srav(Register rd, Register rt, Register rs) {
jeffhao7fbee072012-08-24 17:56:54 -0700400 EmitR(0, rs, rt, rd, 0, 0x07);
401}
402
403void MipsAssembler::Lb(Register rt, Register rs, uint16_t imm16) {
404 EmitI(0x20, rs, rt, imm16);
405}
406
407void MipsAssembler::Lh(Register rt, Register rs, uint16_t imm16) {
408 EmitI(0x21, rs, rt, imm16);
409}
410
411void MipsAssembler::Lw(Register rt, Register rs, uint16_t imm16) {
412 EmitI(0x23, rs, rt, imm16);
413}
414
415void MipsAssembler::Lbu(Register rt, Register rs, uint16_t imm16) {
416 EmitI(0x24, rs, rt, imm16);
417}
418
419void MipsAssembler::Lhu(Register rt, Register rs, uint16_t imm16) {
420 EmitI(0x25, rs, rt, imm16);
421}
422
423void MipsAssembler::Lui(Register rt, uint16_t imm16) {
424 EmitI(0xf, static_cast<Register>(0), rt, imm16);
425}
426
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200427void MipsAssembler::Sync(uint32_t stype) {
428 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0),
429 stype & 0x1f, 0xf);
430}
431
jeffhao7fbee072012-08-24 17:56:54 -0700432void MipsAssembler::Mfhi(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200433 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700434 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x10);
435}
436
437void MipsAssembler::Mflo(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200438 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700439 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x12);
440}
441
442void MipsAssembler::Sb(Register rt, Register rs, uint16_t imm16) {
443 EmitI(0x28, rs, rt, imm16);
444}
445
446void MipsAssembler::Sh(Register rt, Register rs, uint16_t imm16) {
447 EmitI(0x29, rs, rt, imm16);
448}
449
450void MipsAssembler::Sw(Register rt, Register rs, uint16_t imm16) {
451 EmitI(0x2b, rs, rt, imm16);
452}
453
454void MipsAssembler::Slt(Register rd, Register rs, Register rt) {
455 EmitR(0, rs, rt, rd, 0, 0x2a);
456}
457
458void MipsAssembler::Sltu(Register rd, Register rs, Register rt) {
459 EmitR(0, rs, rt, rd, 0, 0x2b);
460}
461
462void MipsAssembler::Slti(Register rt, Register rs, uint16_t imm16) {
463 EmitI(0xa, rs, rt, imm16);
464}
465
466void MipsAssembler::Sltiu(Register rt, Register rs, uint16_t imm16) {
467 EmitI(0xb, rs, rt, imm16);
468}
469
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200470void MipsAssembler::B(uint16_t imm16) {
471 EmitI(0x4, static_cast<Register>(0), static_cast<Register>(0), imm16);
472}
473
474void MipsAssembler::Beq(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700475 EmitI(0x4, rs, rt, imm16);
476}
477
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200478void MipsAssembler::Bne(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700479 EmitI(0x5, rs, rt, imm16);
480}
481
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200482void MipsAssembler::Beqz(Register rt, uint16_t imm16) {
483 Beq(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700484}
485
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200486void MipsAssembler::Bnez(Register rt, uint16_t imm16) {
487 Bne(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700488}
489
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200490void MipsAssembler::Bltz(Register rt, uint16_t imm16) {
491 EmitI(0x1, rt, static_cast<Register>(0), imm16);
492}
493
494void MipsAssembler::Bgez(Register rt, uint16_t imm16) {
495 EmitI(0x1, rt, static_cast<Register>(0x1), imm16);
496}
497
498void MipsAssembler::Blez(Register rt, uint16_t imm16) {
499 EmitI(0x6, rt, static_cast<Register>(0), imm16);
500}
501
502void MipsAssembler::Bgtz(Register rt, uint16_t imm16) {
503 EmitI(0x7, rt, static_cast<Register>(0), imm16);
504}
505
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800506void MipsAssembler::Bc1f(int cc, uint16_t imm16) {
507 CHECK(!IsR6());
508 CHECK(IsUint<3>(cc)) << cc;
509 EmitI(0x11, static_cast<Register>(0x8), static_cast<Register>(cc << 2), imm16);
510}
511
512void MipsAssembler::Bc1t(int cc, uint16_t imm16) {
513 CHECK(!IsR6());
514 CHECK(IsUint<3>(cc)) << cc;
515 EmitI(0x11, static_cast<Register>(0x8), static_cast<Register>((cc << 2) | 1), imm16);
516}
517
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200518void MipsAssembler::J(uint32_t addr26) {
519 EmitI26(0x2, addr26);
520}
521
522void MipsAssembler::Jal(uint32_t addr26) {
523 EmitI26(0x3, addr26);
524}
525
526void MipsAssembler::Jalr(Register rd, Register rs) {
527 EmitR(0, rs, static_cast<Register>(0), rd, 0, 0x09);
jeffhao7fbee072012-08-24 17:56:54 -0700528}
529
530void MipsAssembler::Jalr(Register rs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200531 Jalr(RA, rs);
532}
533
534void MipsAssembler::Jr(Register rs) {
535 Jalr(ZERO, rs);
536}
537
538void MipsAssembler::Nal() {
539 EmitI(0x1, static_cast<Register>(0), static_cast<Register>(0x10), 0);
540}
541
542void MipsAssembler::Auipc(Register rs, uint16_t imm16) {
543 CHECK(IsR6());
544 EmitI(0x3B, rs, static_cast<Register>(0x1E), imm16);
545}
546
547void MipsAssembler::Addiupc(Register rs, uint32_t imm19) {
548 CHECK(IsR6());
549 CHECK(IsUint<19>(imm19)) << imm19;
550 EmitI21(0x3B, rs, imm19);
551}
552
553void MipsAssembler::Bc(uint32_t imm26) {
554 CHECK(IsR6());
555 EmitI26(0x32, imm26);
556}
557
558void MipsAssembler::Jic(Register rt, uint16_t imm16) {
559 CHECK(IsR6());
560 EmitI(0x36, static_cast<Register>(0), rt, imm16);
561}
562
563void MipsAssembler::Jialc(Register rt, uint16_t imm16) {
564 CHECK(IsR6());
565 EmitI(0x3E, static_cast<Register>(0), rt, imm16);
566}
567
568void MipsAssembler::Bltc(Register rs, Register rt, uint16_t imm16) {
569 CHECK(IsR6());
570 CHECK_NE(rs, ZERO);
571 CHECK_NE(rt, ZERO);
572 CHECK_NE(rs, rt);
573 EmitI(0x17, rs, rt, imm16);
574}
575
576void MipsAssembler::Bltzc(Register rt, uint16_t imm16) {
577 CHECK(IsR6());
578 CHECK_NE(rt, ZERO);
579 EmitI(0x17, rt, rt, imm16);
580}
581
582void MipsAssembler::Bgtzc(Register rt, uint16_t imm16) {
583 CHECK(IsR6());
584 CHECK_NE(rt, ZERO);
585 EmitI(0x17, static_cast<Register>(0), rt, imm16);
586}
587
588void MipsAssembler::Bgec(Register rs, Register rt, uint16_t imm16) {
589 CHECK(IsR6());
590 CHECK_NE(rs, ZERO);
591 CHECK_NE(rt, ZERO);
592 CHECK_NE(rs, rt);
593 EmitI(0x16, rs, rt, imm16);
594}
595
596void MipsAssembler::Bgezc(Register rt, uint16_t imm16) {
597 CHECK(IsR6());
598 CHECK_NE(rt, ZERO);
599 EmitI(0x16, rt, rt, imm16);
600}
601
602void MipsAssembler::Blezc(Register rt, uint16_t imm16) {
603 CHECK(IsR6());
604 CHECK_NE(rt, ZERO);
605 EmitI(0x16, static_cast<Register>(0), rt, imm16);
606}
607
608void MipsAssembler::Bltuc(Register rs, Register rt, uint16_t imm16) {
609 CHECK(IsR6());
610 CHECK_NE(rs, ZERO);
611 CHECK_NE(rt, ZERO);
612 CHECK_NE(rs, rt);
613 EmitI(0x7, rs, rt, imm16);
614}
615
616void MipsAssembler::Bgeuc(Register rs, Register rt, uint16_t imm16) {
617 CHECK(IsR6());
618 CHECK_NE(rs, ZERO);
619 CHECK_NE(rt, ZERO);
620 CHECK_NE(rs, rt);
621 EmitI(0x6, rs, rt, imm16);
622}
623
624void MipsAssembler::Beqc(Register rs, Register rt, uint16_t imm16) {
625 CHECK(IsR6());
626 CHECK_NE(rs, ZERO);
627 CHECK_NE(rt, ZERO);
628 CHECK_NE(rs, rt);
629 EmitI(0x8, std::min(rs, rt), std::max(rs, rt), imm16);
630}
631
632void MipsAssembler::Bnec(Register rs, Register rt, uint16_t imm16) {
633 CHECK(IsR6());
634 CHECK_NE(rs, ZERO);
635 CHECK_NE(rt, ZERO);
636 CHECK_NE(rs, rt);
637 EmitI(0x18, std::min(rs, rt), std::max(rs, rt), imm16);
638}
639
640void MipsAssembler::Beqzc(Register rs, uint32_t imm21) {
641 CHECK(IsR6());
642 CHECK_NE(rs, ZERO);
643 EmitI21(0x36, rs, imm21);
644}
645
646void MipsAssembler::Bnezc(Register rs, uint32_t imm21) {
647 CHECK(IsR6());
648 CHECK_NE(rs, ZERO);
649 EmitI21(0x3E, rs, imm21);
650}
651
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800652void MipsAssembler::Bc1eqz(FRegister ft, uint16_t imm16) {
653 CHECK(IsR6());
654 EmitFI(0x11, 0x9, ft, imm16);
655}
656
657void MipsAssembler::Bc1nez(FRegister ft, uint16_t imm16) {
658 CHECK(IsR6());
659 EmitFI(0x11, 0xD, ft, imm16);
660}
661
662void MipsAssembler::EmitBcondR2(BranchCondition cond, Register rs, Register rt, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200663 switch (cond) {
664 case kCondLTZ:
665 CHECK_EQ(rt, ZERO);
666 Bltz(rs, imm16);
667 break;
668 case kCondGEZ:
669 CHECK_EQ(rt, ZERO);
670 Bgez(rs, imm16);
671 break;
672 case kCondLEZ:
673 CHECK_EQ(rt, ZERO);
674 Blez(rs, imm16);
675 break;
676 case kCondGTZ:
677 CHECK_EQ(rt, ZERO);
678 Bgtz(rs, imm16);
679 break;
680 case kCondEQ:
681 Beq(rs, rt, imm16);
682 break;
683 case kCondNE:
684 Bne(rs, rt, imm16);
685 break;
686 case kCondEQZ:
687 CHECK_EQ(rt, ZERO);
688 Beqz(rs, imm16);
689 break;
690 case kCondNEZ:
691 CHECK_EQ(rt, ZERO);
692 Bnez(rs, imm16);
693 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800694 case kCondF:
695 CHECK_EQ(rt, ZERO);
696 Bc1f(static_cast<int>(rs), imm16);
697 break;
698 case kCondT:
699 CHECK_EQ(rt, ZERO);
700 Bc1t(static_cast<int>(rs), imm16);
701 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200702 case kCondLT:
703 case kCondGE:
704 case kCondLE:
705 case kCondGT:
706 case kCondLTU:
707 case kCondGEU:
708 case kUncond:
709 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
710 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
711 LOG(FATAL) << "Unexpected branch condition " << cond;
712 UNREACHABLE();
713 }
714}
715
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800716void MipsAssembler::EmitBcondR6(BranchCondition cond, Register rs, Register rt, uint32_t imm16_21) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200717 switch (cond) {
718 case kCondLT:
719 Bltc(rs, rt, imm16_21);
720 break;
721 case kCondGE:
722 Bgec(rs, rt, imm16_21);
723 break;
724 case kCondLE:
725 Bgec(rt, rs, imm16_21);
726 break;
727 case kCondGT:
728 Bltc(rt, rs, imm16_21);
729 break;
730 case kCondLTZ:
731 CHECK_EQ(rt, ZERO);
732 Bltzc(rs, imm16_21);
733 break;
734 case kCondGEZ:
735 CHECK_EQ(rt, ZERO);
736 Bgezc(rs, imm16_21);
737 break;
738 case kCondLEZ:
739 CHECK_EQ(rt, ZERO);
740 Blezc(rs, imm16_21);
741 break;
742 case kCondGTZ:
743 CHECK_EQ(rt, ZERO);
744 Bgtzc(rs, imm16_21);
745 break;
746 case kCondEQ:
747 Beqc(rs, rt, imm16_21);
748 break;
749 case kCondNE:
750 Bnec(rs, rt, imm16_21);
751 break;
752 case kCondEQZ:
753 CHECK_EQ(rt, ZERO);
754 Beqzc(rs, imm16_21);
755 break;
756 case kCondNEZ:
757 CHECK_EQ(rt, ZERO);
758 Bnezc(rs, imm16_21);
759 break;
760 case kCondLTU:
761 Bltuc(rs, rt, imm16_21);
762 break;
763 case kCondGEU:
764 Bgeuc(rs, rt, imm16_21);
765 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800766 case kCondF:
767 CHECK_EQ(rt, ZERO);
768 Bc1eqz(static_cast<FRegister>(rs), imm16_21);
769 break;
770 case kCondT:
771 CHECK_EQ(rt, ZERO);
772 Bc1nez(static_cast<FRegister>(rs), imm16_21);
773 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200774 case kUncond:
775 LOG(FATAL) << "Unexpected branch condition " << cond;
776 UNREACHABLE();
777 }
jeffhao7fbee072012-08-24 17:56:54 -0700778}
779
780void MipsAssembler::AddS(FRegister fd, FRegister fs, FRegister ft) {
781 EmitFR(0x11, 0x10, ft, fs, fd, 0x0);
782}
783
784void MipsAssembler::SubS(FRegister fd, FRegister fs, FRegister ft) {
785 EmitFR(0x11, 0x10, ft, fs, fd, 0x1);
786}
787
788void MipsAssembler::MulS(FRegister fd, FRegister fs, FRegister ft) {
789 EmitFR(0x11, 0x10, ft, fs, fd, 0x2);
790}
791
792void MipsAssembler::DivS(FRegister fd, FRegister fs, FRegister ft) {
793 EmitFR(0x11, 0x10, ft, fs, fd, 0x3);
794}
795
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200796void MipsAssembler::AddD(FRegister fd, FRegister fs, FRegister ft) {
797 EmitFR(0x11, 0x11, ft, fs, fd, 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700798}
799
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200800void MipsAssembler::SubD(FRegister fd, FRegister fs, FRegister ft) {
801 EmitFR(0x11, 0x11, ft, fs, fd, 0x1);
jeffhao7fbee072012-08-24 17:56:54 -0700802}
803
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200804void MipsAssembler::MulD(FRegister fd, FRegister fs, FRegister ft) {
805 EmitFR(0x11, 0x11, ft, fs, fd, 0x2);
jeffhao7fbee072012-08-24 17:56:54 -0700806}
807
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200808void MipsAssembler::DivD(FRegister fd, FRegister fs, FRegister ft) {
809 EmitFR(0x11, 0x11, ft, fs, fd, 0x3);
jeffhao7fbee072012-08-24 17:56:54 -0700810}
811
812void MipsAssembler::MovS(FRegister fd, FRegister fs) {
813 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x6);
814}
815
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200816void MipsAssembler::MovD(FRegister fd, FRegister fs) {
817 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x6);
818}
819
820void MipsAssembler::NegS(FRegister fd, FRegister fs) {
821 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x7);
822}
823
824void MipsAssembler::NegD(FRegister fd, FRegister fs) {
825 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x7);
826}
827
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800828void MipsAssembler::CunS(int cc, FRegister fs, FRegister ft) {
829 CHECK(!IsR6());
830 CHECK(IsUint<3>(cc)) << cc;
831 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x31);
832}
833
834void MipsAssembler::CeqS(int cc, FRegister fs, FRegister ft) {
835 CHECK(!IsR6());
836 CHECK(IsUint<3>(cc)) << cc;
837 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x32);
838}
839
840void MipsAssembler::CueqS(int cc, FRegister fs, FRegister ft) {
841 CHECK(!IsR6());
842 CHECK(IsUint<3>(cc)) << cc;
843 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x33);
844}
845
846void MipsAssembler::ColtS(int cc, FRegister fs, FRegister ft) {
847 CHECK(!IsR6());
848 CHECK(IsUint<3>(cc)) << cc;
849 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x34);
850}
851
852void MipsAssembler::CultS(int cc, FRegister fs, FRegister ft) {
853 CHECK(!IsR6());
854 CHECK(IsUint<3>(cc)) << cc;
855 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x35);
856}
857
858void MipsAssembler::ColeS(int cc, FRegister fs, FRegister ft) {
859 CHECK(!IsR6());
860 CHECK(IsUint<3>(cc)) << cc;
861 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x36);
862}
863
864void MipsAssembler::CuleS(int cc, FRegister fs, FRegister ft) {
865 CHECK(!IsR6());
866 CHECK(IsUint<3>(cc)) << cc;
867 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x37);
868}
869
870void MipsAssembler::CunD(int cc, FRegister fs, FRegister ft) {
871 CHECK(!IsR6());
872 CHECK(IsUint<3>(cc)) << cc;
873 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x31);
874}
875
876void MipsAssembler::CeqD(int cc, FRegister fs, FRegister ft) {
877 CHECK(!IsR6());
878 CHECK(IsUint<3>(cc)) << cc;
879 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x32);
880}
881
882void MipsAssembler::CueqD(int cc, FRegister fs, FRegister ft) {
883 CHECK(!IsR6());
884 CHECK(IsUint<3>(cc)) << cc;
885 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x33);
886}
887
888void MipsAssembler::ColtD(int cc, FRegister fs, FRegister ft) {
889 CHECK(!IsR6());
890 CHECK(IsUint<3>(cc)) << cc;
891 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x34);
892}
893
894void MipsAssembler::CultD(int cc, FRegister fs, FRegister ft) {
895 CHECK(!IsR6());
896 CHECK(IsUint<3>(cc)) << cc;
897 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x35);
898}
899
900void MipsAssembler::ColeD(int cc, FRegister fs, FRegister ft) {
901 CHECK(!IsR6());
902 CHECK(IsUint<3>(cc)) << cc;
903 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x36);
904}
905
906void MipsAssembler::CuleD(int cc, FRegister fs, FRegister ft) {
907 CHECK(!IsR6());
908 CHECK(IsUint<3>(cc)) << cc;
909 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x37);
910}
911
912void MipsAssembler::CmpUnS(FRegister fd, FRegister fs, FRegister ft) {
913 CHECK(IsR6());
914 EmitFR(0x11, 0x14, ft, fs, fd, 0x01);
915}
916
917void MipsAssembler::CmpEqS(FRegister fd, FRegister fs, FRegister ft) {
918 CHECK(IsR6());
919 EmitFR(0x11, 0x14, ft, fs, fd, 0x02);
920}
921
922void MipsAssembler::CmpUeqS(FRegister fd, FRegister fs, FRegister ft) {
923 CHECK(IsR6());
924 EmitFR(0x11, 0x14, ft, fs, fd, 0x03);
925}
926
927void MipsAssembler::CmpLtS(FRegister fd, FRegister fs, FRegister ft) {
928 CHECK(IsR6());
929 EmitFR(0x11, 0x14, ft, fs, fd, 0x04);
930}
931
932void MipsAssembler::CmpUltS(FRegister fd, FRegister fs, FRegister ft) {
933 CHECK(IsR6());
934 EmitFR(0x11, 0x14, ft, fs, fd, 0x05);
935}
936
937void MipsAssembler::CmpLeS(FRegister fd, FRegister fs, FRegister ft) {
938 CHECK(IsR6());
939 EmitFR(0x11, 0x14, ft, fs, fd, 0x06);
940}
941
942void MipsAssembler::CmpUleS(FRegister fd, FRegister fs, FRegister ft) {
943 CHECK(IsR6());
944 EmitFR(0x11, 0x14, ft, fs, fd, 0x07);
945}
946
947void MipsAssembler::CmpOrS(FRegister fd, FRegister fs, FRegister ft) {
948 CHECK(IsR6());
949 EmitFR(0x11, 0x14, ft, fs, fd, 0x11);
950}
951
952void MipsAssembler::CmpUneS(FRegister fd, FRegister fs, FRegister ft) {
953 CHECK(IsR6());
954 EmitFR(0x11, 0x14, ft, fs, fd, 0x12);
955}
956
957void MipsAssembler::CmpNeS(FRegister fd, FRegister fs, FRegister ft) {
958 CHECK(IsR6());
959 EmitFR(0x11, 0x14, ft, fs, fd, 0x13);
960}
961
962void MipsAssembler::CmpUnD(FRegister fd, FRegister fs, FRegister ft) {
963 CHECK(IsR6());
964 EmitFR(0x11, 0x15, ft, fs, fd, 0x01);
965}
966
967void MipsAssembler::CmpEqD(FRegister fd, FRegister fs, FRegister ft) {
968 CHECK(IsR6());
969 EmitFR(0x11, 0x15, ft, fs, fd, 0x02);
970}
971
972void MipsAssembler::CmpUeqD(FRegister fd, FRegister fs, FRegister ft) {
973 CHECK(IsR6());
974 EmitFR(0x11, 0x15, ft, fs, fd, 0x03);
975}
976
977void MipsAssembler::CmpLtD(FRegister fd, FRegister fs, FRegister ft) {
978 CHECK(IsR6());
979 EmitFR(0x11, 0x15, ft, fs, fd, 0x04);
980}
981
982void MipsAssembler::CmpUltD(FRegister fd, FRegister fs, FRegister ft) {
983 CHECK(IsR6());
984 EmitFR(0x11, 0x15, ft, fs, fd, 0x05);
985}
986
987void MipsAssembler::CmpLeD(FRegister fd, FRegister fs, FRegister ft) {
988 CHECK(IsR6());
989 EmitFR(0x11, 0x15, ft, fs, fd, 0x06);
990}
991
992void MipsAssembler::CmpUleD(FRegister fd, FRegister fs, FRegister ft) {
993 CHECK(IsR6());
994 EmitFR(0x11, 0x15, ft, fs, fd, 0x07);
995}
996
997void MipsAssembler::CmpOrD(FRegister fd, FRegister fs, FRegister ft) {
998 CHECK(IsR6());
999 EmitFR(0x11, 0x15, ft, fs, fd, 0x11);
1000}
1001
1002void MipsAssembler::CmpUneD(FRegister fd, FRegister fs, FRegister ft) {
1003 CHECK(IsR6());
1004 EmitFR(0x11, 0x15, ft, fs, fd, 0x12);
1005}
1006
1007void MipsAssembler::CmpNeD(FRegister fd, FRegister fs, FRegister ft) {
1008 CHECK(IsR6());
1009 EmitFR(0x11, 0x15, ft, fs, fd, 0x13);
1010}
1011
1012void MipsAssembler::Movf(Register rd, Register rs, int cc) {
1013 CHECK(!IsR6());
1014 CHECK(IsUint<3>(cc)) << cc;
1015 EmitR(0, rs, static_cast<Register>(cc << 2), rd, 0, 0x01);
1016}
1017
1018void MipsAssembler::Movt(Register rd, Register rs, int cc) {
1019 CHECK(!IsR6());
1020 CHECK(IsUint<3>(cc)) << cc;
1021 EmitR(0, rs, static_cast<Register>((cc << 2) | 1), rd, 0, 0x01);
1022}
1023
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001024void MipsAssembler::Cvtsw(FRegister fd, FRegister fs) {
1025 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x20);
1026}
1027
1028void MipsAssembler::Cvtdw(FRegister fd, FRegister fs) {
1029 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x21);
1030}
1031
1032void MipsAssembler::Cvtsd(FRegister fd, FRegister fs) {
1033 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x20);
1034}
1035
1036void MipsAssembler::Cvtds(FRegister fd, FRegister fs) {
1037 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x21);
jeffhao7fbee072012-08-24 17:56:54 -07001038}
1039
1040void MipsAssembler::Mfc1(Register rt, FRegister fs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001041 EmitFR(0x11, 0x00, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -07001042}
1043
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001044void MipsAssembler::Mtc1(Register rt, FRegister fs) {
1045 EmitFR(0x11, 0x04, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
1046}
1047
1048void MipsAssembler::Mfhc1(Register rt, FRegister fs) {
1049 EmitFR(0x11, 0x03, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
1050}
1051
1052void MipsAssembler::Mthc1(Register rt, FRegister fs) {
1053 EmitFR(0x11, 0x07, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -07001054}
1055
1056void MipsAssembler::Lwc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001057 EmitI(0x31, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001058}
1059
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001060void MipsAssembler::Ldc1(FRegister ft, Register rs, uint16_t imm16) {
1061 EmitI(0x35, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001062}
1063
1064void MipsAssembler::Swc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001065 EmitI(0x39, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001066}
1067
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001068void MipsAssembler::Sdc1(FRegister ft, Register rs, uint16_t imm16) {
1069 EmitI(0x3d, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001070}
1071
1072void MipsAssembler::Break() {
1073 EmitR(0, static_cast<Register>(0), static_cast<Register>(0),
1074 static_cast<Register>(0), 0, 0xD);
1075}
1076
jeffhao07030602012-09-26 14:33:14 -07001077void MipsAssembler::Nop() {
1078 EmitR(0x0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0), 0, 0x0);
1079}
1080
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001081void MipsAssembler::Move(Register rd, Register rs) {
1082 Or(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001083}
1084
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001085void MipsAssembler::Clear(Register rd) {
1086 Move(rd, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001087}
1088
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001089void MipsAssembler::Not(Register rd, Register rs) {
1090 Nor(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001091}
1092
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001093void MipsAssembler::Push(Register rs) {
1094 IncreaseFrameSize(kMipsWordSize);
1095 Sw(rs, SP, 0);
jeffhao7fbee072012-08-24 17:56:54 -07001096}
1097
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001098void MipsAssembler::Pop(Register rd) {
1099 Lw(rd, SP, 0);
1100 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001101}
1102
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001103void MipsAssembler::PopAndReturn(Register rd, Register rt) {
1104 Lw(rd, SP, 0);
1105 Jr(rt);
1106 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001107}
1108
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001109void MipsAssembler::LoadConst32(Register rd, int32_t value) {
1110 if (IsUint<16>(value)) {
1111 // Use OR with (unsigned) immediate to encode 16b unsigned int.
1112 Ori(rd, ZERO, value);
1113 } else if (IsInt<16>(value)) {
1114 // Use ADD with (signed) immediate to encode 16b signed int.
1115 Addiu(rd, ZERO, value);
jeffhao7fbee072012-08-24 17:56:54 -07001116 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001117 Lui(rd, High16Bits(value));
1118 if (value & 0xFFFF)
1119 Ori(rd, rd, Low16Bits(value));
1120 }
1121}
1122
1123void MipsAssembler::LoadConst64(Register reg_hi, Register reg_lo, int64_t value) {
1124 LoadConst32(reg_lo, Low32Bits(value));
1125 LoadConst32(reg_hi, High32Bits(value));
1126}
1127
1128void MipsAssembler::StoreConst32ToOffset(int32_t value,
1129 Register base,
1130 int32_t offset,
1131 Register temp) {
1132 if (!IsInt<16>(offset)) {
1133 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
1134 LoadConst32(AT, offset);
1135 Addu(AT, AT, base);
1136 base = AT;
1137 offset = 0;
1138 }
1139 LoadConst32(temp, value);
1140 Sw(temp, base, offset);
1141}
1142
1143void MipsAssembler::StoreConst64ToOffset(int64_t value,
1144 Register base,
1145 int32_t offset,
1146 Register temp) {
1147 // IsInt<16> must be passed a signed value.
1148 if (!IsInt<16>(offset) || !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize))) {
1149 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
1150 LoadConst32(AT, offset);
1151 Addu(AT, AT, base);
1152 base = AT;
1153 offset = 0;
1154 }
1155 LoadConst32(temp, Low32Bits(value));
1156 Sw(temp, base, offset);
1157 LoadConst32(temp, High32Bits(value));
1158 Sw(temp, base, offset + kMipsWordSize);
1159}
1160
1161void MipsAssembler::LoadSConst32(FRegister r, int32_t value, Register temp) {
1162 LoadConst32(temp, value);
1163 Mtc1(temp, r);
1164}
1165
1166void MipsAssembler::LoadDConst64(FRegister rd, int64_t value, Register temp) {
1167 LoadConst32(temp, Low32Bits(value));
1168 Mtc1(temp, rd);
1169 LoadConst32(temp, High32Bits(value));
1170 Mthc1(temp, rd);
1171}
1172
1173void MipsAssembler::Addiu32(Register rt, Register rs, int32_t value, Register temp) {
1174 if (IsInt<16>(value)) {
1175 Addiu(rt, rs, value);
1176 } else {
1177 LoadConst32(temp, value);
1178 Addu(rt, rs, temp);
1179 }
1180}
1181
1182void MipsAssembler::Branch::InitShortOrLong(MipsAssembler::Branch::OffsetBits offset_size,
1183 MipsAssembler::Branch::Type short_type,
1184 MipsAssembler::Branch::Type long_type) {
1185 type_ = (offset_size <= branch_info_[short_type].offset_size) ? short_type : long_type;
1186}
1187
1188void MipsAssembler::Branch::InitializeType(bool is_call, bool is_r6) {
1189 OffsetBits offset_size = GetOffsetSizeNeeded(location_, target_);
1190 if (is_r6) {
1191 // R6
1192 if (is_call) {
1193 InitShortOrLong(offset_size, kR6Call, kR6LongCall);
1194 } else if (condition_ == kUncond) {
1195 InitShortOrLong(offset_size, kR6UncondBranch, kR6LongUncondBranch);
1196 } else {
1197 if (condition_ == kCondEQZ || condition_ == kCondNEZ) {
1198 // Special case for beqzc/bnezc with longer offset than in other b<cond>c instructions.
1199 type_ = (offset_size <= kOffset23) ? kR6CondBranch : kR6LongCondBranch;
1200 } else {
1201 InitShortOrLong(offset_size, kR6CondBranch, kR6LongCondBranch);
1202 }
1203 }
1204 } else {
1205 // R2
1206 if (is_call) {
1207 InitShortOrLong(offset_size, kCall, kLongCall);
1208 } else if (condition_ == kUncond) {
1209 InitShortOrLong(offset_size, kUncondBranch, kLongUncondBranch);
1210 } else {
1211 InitShortOrLong(offset_size, kCondBranch, kLongCondBranch);
1212 }
1213 }
1214 old_type_ = type_;
1215}
1216
1217bool MipsAssembler::Branch::IsNop(BranchCondition condition, Register lhs, Register rhs) {
1218 switch (condition) {
1219 case kCondLT:
1220 case kCondGT:
1221 case kCondNE:
1222 case kCondLTU:
1223 return lhs == rhs;
1224 default:
1225 return false;
1226 }
1227}
1228
1229bool MipsAssembler::Branch::IsUncond(BranchCondition condition, Register lhs, Register rhs) {
1230 switch (condition) {
1231 case kUncond:
1232 return true;
1233 case kCondGE:
1234 case kCondLE:
1235 case kCondEQ:
1236 case kCondGEU:
1237 return lhs == rhs;
1238 default:
1239 return false;
1240 }
1241}
1242
1243MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target)
1244 : old_location_(location),
1245 location_(location),
1246 target_(target),
1247 lhs_reg_(0),
1248 rhs_reg_(0),
1249 condition_(kUncond) {
1250 InitializeType(false, is_r6);
1251}
1252
1253MipsAssembler::Branch::Branch(bool is_r6,
1254 uint32_t location,
1255 uint32_t target,
1256 MipsAssembler::BranchCondition condition,
1257 Register lhs_reg,
1258 Register rhs_reg)
1259 : old_location_(location),
1260 location_(location),
1261 target_(target),
1262 lhs_reg_(lhs_reg),
1263 rhs_reg_(rhs_reg),
1264 condition_(condition) {
1265 CHECK_NE(condition, kUncond);
1266 switch (condition) {
1267 case kCondLT:
1268 case kCondGE:
1269 case kCondLE:
1270 case kCondGT:
1271 case kCondLTU:
1272 case kCondGEU:
1273 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
1274 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
1275 // We leave this up to the caller.
1276 CHECK(is_r6);
1277 FALLTHROUGH_INTENDED;
1278 case kCondEQ:
1279 case kCondNE:
1280 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1281 // To compare with 0, use dedicated kCond*Z conditions.
1282 CHECK_NE(lhs_reg, ZERO);
1283 CHECK_NE(rhs_reg, ZERO);
1284 break;
1285 case kCondLTZ:
1286 case kCondGEZ:
1287 case kCondLEZ:
1288 case kCondGTZ:
1289 case kCondEQZ:
1290 case kCondNEZ:
1291 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1292 CHECK_NE(lhs_reg, ZERO);
1293 CHECK_EQ(rhs_reg, ZERO);
1294 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001295 case kCondF:
1296 case kCondT:
1297 CHECK_EQ(rhs_reg, ZERO);
1298 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001299 case kUncond:
1300 UNREACHABLE();
1301 }
1302 CHECK(!IsNop(condition, lhs_reg, rhs_reg));
1303 if (IsUncond(condition, lhs_reg, rhs_reg)) {
1304 // Branch condition is always true, make the branch unconditional.
1305 condition_ = kUncond;
1306 }
1307 InitializeType(false, is_r6);
1308}
1309
1310MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target, Register indirect_reg)
1311 : old_location_(location),
1312 location_(location),
1313 target_(target),
1314 lhs_reg_(indirect_reg),
1315 rhs_reg_(0),
1316 condition_(kUncond) {
1317 CHECK_NE(indirect_reg, ZERO);
1318 CHECK_NE(indirect_reg, AT);
1319 InitializeType(true, is_r6);
1320}
1321
1322MipsAssembler::BranchCondition MipsAssembler::Branch::OppositeCondition(
1323 MipsAssembler::BranchCondition cond) {
1324 switch (cond) {
1325 case kCondLT:
1326 return kCondGE;
1327 case kCondGE:
1328 return kCondLT;
1329 case kCondLE:
1330 return kCondGT;
1331 case kCondGT:
1332 return kCondLE;
1333 case kCondLTZ:
1334 return kCondGEZ;
1335 case kCondGEZ:
1336 return kCondLTZ;
1337 case kCondLEZ:
1338 return kCondGTZ;
1339 case kCondGTZ:
1340 return kCondLEZ;
1341 case kCondEQ:
1342 return kCondNE;
1343 case kCondNE:
1344 return kCondEQ;
1345 case kCondEQZ:
1346 return kCondNEZ;
1347 case kCondNEZ:
1348 return kCondEQZ;
1349 case kCondLTU:
1350 return kCondGEU;
1351 case kCondGEU:
1352 return kCondLTU;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001353 case kCondF:
1354 return kCondT;
1355 case kCondT:
1356 return kCondF;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001357 case kUncond:
1358 LOG(FATAL) << "Unexpected branch condition " << cond;
1359 }
1360 UNREACHABLE();
1361}
1362
1363MipsAssembler::Branch::Type MipsAssembler::Branch::GetType() const {
1364 return type_;
1365}
1366
1367MipsAssembler::BranchCondition MipsAssembler::Branch::GetCondition() const {
1368 return condition_;
1369}
1370
1371Register MipsAssembler::Branch::GetLeftRegister() const {
1372 return static_cast<Register>(lhs_reg_);
1373}
1374
1375Register MipsAssembler::Branch::GetRightRegister() const {
1376 return static_cast<Register>(rhs_reg_);
1377}
1378
1379uint32_t MipsAssembler::Branch::GetTarget() const {
1380 return target_;
1381}
1382
1383uint32_t MipsAssembler::Branch::GetLocation() const {
1384 return location_;
1385}
1386
1387uint32_t MipsAssembler::Branch::GetOldLocation() const {
1388 return old_location_;
1389}
1390
1391uint32_t MipsAssembler::Branch::GetLength() const {
1392 return branch_info_[type_].length;
1393}
1394
1395uint32_t MipsAssembler::Branch::GetOldLength() const {
1396 return branch_info_[old_type_].length;
1397}
1398
1399uint32_t MipsAssembler::Branch::GetSize() const {
1400 return GetLength() * sizeof(uint32_t);
1401}
1402
1403uint32_t MipsAssembler::Branch::GetOldSize() const {
1404 return GetOldLength() * sizeof(uint32_t);
1405}
1406
1407uint32_t MipsAssembler::Branch::GetEndLocation() const {
1408 return GetLocation() + GetSize();
1409}
1410
1411uint32_t MipsAssembler::Branch::GetOldEndLocation() const {
1412 return GetOldLocation() + GetOldSize();
1413}
1414
1415bool MipsAssembler::Branch::IsLong() const {
1416 switch (type_) {
1417 // R2 short branches.
1418 case kUncondBranch:
1419 case kCondBranch:
1420 case kCall:
1421 // R6 short branches.
1422 case kR6UncondBranch:
1423 case kR6CondBranch:
1424 case kR6Call:
1425 return false;
1426 // R2 long branches.
1427 case kLongUncondBranch:
1428 case kLongCondBranch:
1429 case kLongCall:
1430 // R6 long branches.
1431 case kR6LongUncondBranch:
1432 case kR6LongCondBranch:
1433 case kR6LongCall:
1434 return true;
1435 }
1436 UNREACHABLE();
1437}
1438
1439bool MipsAssembler::Branch::IsResolved() const {
1440 return target_ != kUnresolved;
1441}
1442
1443MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSize() const {
1444 OffsetBits offset_size =
1445 (type_ == kR6CondBranch && (condition_ == kCondEQZ || condition_ == kCondNEZ))
1446 ? kOffset23
1447 : branch_info_[type_].offset_size;
1448 return offset_size;
1449}
1450
1451MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSizeNeeded(uint32_t location,
1452 uint32_t target) {
1453 // For unresolved targets assume the shortest encoding
1454 // (later it will be made longer if needed).
1455 if (target == kUnresolved)
1456 return kOffset16;
1457 int64_t distance = static_cast<int64_t>(target) - location;
1458 // To simplify calculations in composite branches consisting of multiple instructions
1459 // bump up the distance by a value larger than the max byte size of a composite branch.
1460 distance += (distance >= 0) ? kMaxBranchSize : -kMaxBranchSize;
1461 if (IsInt<kOffset16>(distance))
1462 return kOffset16;
1463 else if (IsInt<kOffset18>(distance))
1464 return kOffset18;
1465 else if (IsInt<kOffset21>(distance))
1466 return kOffset21;
1467 else if (IsInt<kOffset23>(distance))
1468 return kOffset23;
1469 else if (IsInt<kOffset28>(distance))
1470 return kOffset28;
1471 return kOffset32;
1472}
1473
1474void MipsAssembler::Branch::Resolve(uint32_t target) {
1475 target_ = target;
1476}
1477
1478void MipsAssembler::Branch::Relocate(uint32_t expand_location, uint32_t delta) {
1479 if (location_ > expand_location) {
1480 location_ += delta;
1481 }
1482 if (!IsResolved()) {
1483 return; // Don't know the target yet.
1484 }
1485 if (target_ > expand_location) {
1486 target_ += delta;
1487 }
1488}
1489
1490void MipsAssembler::Branch::PromoteToLong() {
1491 switch (type_) {
1492 // R2 short branches.
1493 case kUncondBranch:
1494 type_ = kLongUncondBranch;
1495 break;
1496 case kCondBranch:
1497 type_ = kLongCondBranch;
1498 break;
1499 case kCall:
1500 type_ = kLongCall;
1501 break;
1502 // R6 short branches.
1503 case kR6UncondBranch:
1504 type_ = kR6LongUncondBranch;
1505 break;
1506 case kR6CondBranch:
1507 type_ = kR6LongCondBranch;
1508 break;
1509 case kR6Call:
1510 type_ = kR6LongCall;
1511 break;
1512 default:
1513 // Note: 'type_' is already long.
1514 break;
1515 }
1516 CHECK(IsLong());
1517}
1518
1519uint32_t MipsAssembler::Branch::PromoteIfNeeded(uint32_t max_short_distance) {
1520 // If the branch is still unresolved or already long, nothing to do.
1521 if (IsLong() || !IsResolved()) {
1522 return 0;
1523 }
1524 // Promote the short branch to long if the offset size is too small
1525 // to hold the distance between location_ and target_.
1526 if (GetOffsetSizeNeeded(location_, target_) > GetOffsetSize()) {
1527 PromoteToLong();
1528 uint32_t old_size = GetOldSize();
1529 uint32_t new_size = GetSize();
1530 CHECK_GT(new_size, old_size);
1531 return new_size - old_size;
1532 }
1533 // The following logic is for debugging/testing purposes.
1534 // Promote some short branches to long when it's not really required.
1535 if (UNLIKELY(max_short_distance != std::numeric_limits<uint32_t>::max())) {
1536 int64_t distance = static_cast<int64_t>(target_) - location_;
1537 distance = (distance >= 0) ? distance : -distance;
1538 if (distance >= max_short_distance) {
1539 PromoteToLong();
1540 uint32_t old_size = GetOldSize();
1541 uint32_t new_size = GetSize();
1542 CHECK_GT(new_size, old_size);
1543 return new_size - old_size;
1544 }
1545 }
1546 return 0;
1547}
1548
1549uint32_t MipsAssembler::Branch::GetOffsetLocation() const {
1550 return location_ + branch_info_[type_].instr_offset * sizeof(uint32_t);
1551}
1552
1553uint32_t MipsAssembler::Branch::GetOffset() const {
1554 CHECK(IsResolved());
1555 uint32_t ofs_mask = 0xFFFFFFFF >> (32 - GetOffsetSize());
1556 // Calculate the byte distance between instructions and also account for
1557 // different PC-relative origins.
1558 uint32_t offset = target_ - GetOffsetLocation() - branch_info_[type_].pc_org * sizeof(uint32_t);
1559 // Prepare the offset for encoding into the instruction(s).
1560 offset = (offset & ofs_mask) >> branch_info_[type_].offset_shift;
1561 return offset;
1562}
1563
1564MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) {
1565 CHECK_LT(branch_id, branches_.size());
1566 return &branches_[branch_id];
1567}
1568
1569const MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) const {
1570 CHECK_LT(branch_id, branches_.size());
1571 return &branches_[branch_id];
1572}
1573
1574void MipsAssembler::Bind(MipsLabel* label) {
1575 CHECK(!label->IsBound());
1576 uint32_t bound_pc = buffer_.Size();
1577
1578 // Walk the list of branches referring to and preceding this label.
1579 // Store the previously unknown target addresses in them.
1580 while (label->IsLinked()) {
1581 uint32_t branch_id = label->Position();
1582 Branch* branch = GetBranch(branch_id);
1583 branch->Resolve(bound_pc);
1584
1585 uint32_t branch_location = branch->GetLocation();
1586 // Extract the location of the previous branch in the list (walking the list backwards;
1587 // the previous branch ID was stored in the space reserved for this branch).
1588 uint32_t prev = buffer_.Load<uint32_t>(branch_location);
1589
1590 // On to the previous branch in the list...
1591 label->position_ = prev;
1592 }
1593
1594 // Now make the label object contain its own location (relative to the end of the preceding
1595 // branch, if any; it will be used by the branches referring to and following this label).
1596 label->prev_branch_id_plus_one_ = branches_.size();
1597 if (label->prev_branch_id_plus_one_) {
1598 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1599 const Branch* branch = GetBranch(branch_id);
1600 bound_pc -= branch->GetEndLocation();
1601 }
1602 label->BindTo(bound_pc);
1603}
1604
1605uint32_t MipsAssembler::GetLabelLocation(MipsLabel* label) const {
1606 CHECK(label->IsBound());
1607 uint32_t target = label->Position();
1608 if (label->prev_branch_id_plus_one_) {
1609 // Get label location based on the branch preceding it.
1610 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1611 const Branch* branch = GetBranch(branch_id);
1612 target += branch->GetEndLocation();
1613 }
1614 return target;
1615}
1616
1617uint32_t MipsAssembler::GetAdjustedPosition(uint32_t old_position) {
1618 // We can reconstruct the adjustment by going through all the branches from the beginning
1619 // up to the old_position. Since we expect AdjustedPosition() to be called in a loop
1620 // with increasing old_position, we can use the data from last AdjustedPosition() to
1621 // continue where we left off and the whole loop should be O(m+n) where m is the number
1622 // of positions to adjust and n is the number of branches.
1623 if (old_position < last_old_position_) {
1624 last_position_adjustment_ = 0;
1625 last_old_position_ = 0;
1626 last_branch_id_ = 0;
1627 }
1628 while (last_branch_id_ != branches_.size()) {
1629 const Branch* branch = GetBranch(last_branch_id_);
1630 if (branch->GetLocation() >= old_position + last_position_adjustment_) {
1631 break;
1632 }
1633 last_position_adjustment_ += branch->GetSize() - branch->GetOldSize();
1634 ++last_branch_id_;
1635 }
1636 last_old_position_ = old_position;
1637 return old_position + last_position_adjustment_;
1638}
1639
1640void MipsAssembler::FinalizeLabeledBranch(MipsLabel* label) {
1641 uint32_t length = branches_.back().GetLength();
1642 if (!label->IsBound()) {
1643 // Branch forward (to a following label), distance is unknown.
1644 // The first branch forward will contain 0, serving as the terminator of
1645 // the list of forward-reaching branches.
1646 Emit(label->position_);
1647 length--;
1648 // Now make the label object point to this branch
1649 // (this forms a linked list of branches preceding this label).
1650 uint32_t branch_id = branches_.size() - 1;
1651 label->LinkTo(branch_id);
1652 }
1653 // Reserve space for the branch.
1654 while (length--) {
1655 Nop();
1656 }
1657}
1658
1659void MipsAssembler::Buncond(MipsLabel* label) {
1660 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1661 branches_.emplace_back(IsR6(), buffer_.Size(), target);
1662 FinalizeLabeledBranch(label);
1663}
1664
1665void MipsAssembler::Bcond(MipsLabel* label, BranchCondition condition, Register lhs, Register rhs) {
1666 // If lhs = rhs, this can be a NOP.
1667 if (Branch::IsNop(condition, lhs, rhs)) {
1668 return;
1669 }
1670 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1671 branches_.emplace_back(IsR6(), buffer_.Size(), target, condition, lhs, rhs);
1672 FinalizeLabeledBranch(label);
1673}
1674
1675void MipsAssembler::Call(MipsLabel* label, Register indirect_reg) {
1676 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1677 branches_.emplace_back(IsR6(), buffer_.Size(), target, indirect_reg);
1678 FinalizeLabeledBranch(label);
1679}
1680
1681void MipsAssembler::PromoteBranches() {
1682 // Promote short branches to long as necessary.
1683 bool changed;
1684 do {
1685 changed = false;
1686 for (auto& branch : branches_) {
1687 CHECK(branch.IsResolved());
1688 uint32_t delta = branch.PromoteIfNeeded();
1689 // If this branch has been promoted and needs to expand in size,
1690 // relocate all branches by the expansion size.
1691 if (delta) {
1692 changed = true;
1693 uint32_t expand_location = branch.GetLocation();
1694 for (auto& branch2 : branches_) {
1695 branch2.Relocate(expand_location, delta);
1696 }
1697 }
1698 }
1699 } while (changed);
1700
1701 // Account for branch expansion by resizing the code buffer
1702 // and moving the code in it to its final location.
1703 size_t branch_count = branches_.size();
1704 if (branch_count > 0) {
1705 // Resize.
1706 Branch& last_branch = branches_[branch_count - 1];
1707 uint32_t size_delta = last_branch.GetEndLocation() - last_branch.GetOldEndLocation();
1708 uint32_t old_size = buffer_.Size();
1709 buffer_.Resize(old_size + size_delta);
1710 // Move the code residing between branch placeholders.
1711 uint32_t end = old_size;
1712 for (size_t i = branch_count; i > 0; ) {
1713 Branch& branch = branches_[--i];
1714 uint32_t size = end - branch.GetOldEndLocation();
1715 buffer_.Move(branch.GetEndLocation(), branch.GetOldEndLocation(), size);
1716 end = branch.GetOldLocation();
1717 }
1718 }
1719}
1720
1721// Note: make sure branch_info_[] and EmitBranch() are kept synchronized.
1722const MipsAssembler::Branch::BranchInfo MipsAssembler::Branch::branch_info_[] = {
1723 // R2 short branches.
1724 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kUncondBranch
1725 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kCondBranch
1726 { 5, 2, 0, MipsAssembler::Branch::kOffset16, 0 }, // kCall
1727 // R2 long branches.
1728 { 9, 3, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongUncondBranch
1729 { 10, 4, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCondBranch
1730 { 6, 1, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCall
1731 // R6 short branches.
1732 { 1, 0, 1, MipsAssembler::Branch::kOffset28, 2 }, // kR6UncondBranch
1733 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kR6CondBranch
1734 // Exception: kOffset23 for beqzc/bnezc.
1735 { 2, 0, 0, MipsAssembler::Branch::kOffset21, 2 }, // kR6Call
1736 // R6 long branches.
1737 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongUncondBranch
1738 { 3, 1, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCondBranch
1739 { 3, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCall
1740};
1741
1742// Note: make sure branch_info_[] and mitBranch() are kept synchronized.
1743void MipsAssembler::EmitBranch(MipsAssembler::Branch* branch) {
1744 CHECK_EQ(overwriting_, true);
1745 overwrite_location_ = branch->GetLocation();
1746 uint32_t offset = branch->GetOffset();
1747 BranchCondition condition = branch->GetCondition();
1748 Register lhs = branch->GetLeftRegister();
1749 Register rhs = branch->GetRightRegister();
1750 switch (branch->GetType()) {
1751 // R2 short branches.
1752 case Branch::kUncondBranch:
1753 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1754 B(offset);
1755 Nop(); // TODO: improve by filling the delay slot.
1756 break;
1757 case Branch::kCondBranch:
1758 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001759 EmitBcondR2(condition, lhs, rhs, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001760 Nop(); // TODO: improve by filling the delay slot.
1761 break;
1762 case Branch::kCall:
1763 Nal();
1764 Nop(); // TODO: is this NOP really needed here?
1765 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1766 Addiu(lhs, RA, offset);
1767 Jalr(lhs);
1768 Nop();
1769 break;
1770
1771 // R2 long branches.
1772 case Branch::kLongUncondBranch:
1773 // To get the value of the PC register we need to use the NAL instruction.
1774 // NAL clobbers the RA register. However, RA must be preserved if the
1775 // method is compiled without the entry/exit sequences that would take care
1776 // of preserving RA (typically, leaf methods don't preserve RA explicitly).
1777 // So, we need to preserve RA in some temporary storage ourselves. The AT
1778 // register can't be used for this because we need it to load a constant
1779 // which will be added to the value that NAL stores in RA. And we can't
1780 // use T9 for this in the context of the JNI compiler, which uses it
1781 // as a scratch register (see InterproceduralScratchRegister()).
1782 // If we were to add a 32-bit constant to RA using two ADDIU instructions,
1783 // we'd also need to use the ROTR instruction, which requires no less than
1784 // MIPSR2.
1785 // Perhaps, we could use T8 or one of R2's multiplier/divider registers
1786 // (LO or HI) or even a floating-point register, but that doesn't seem
1787 // like a nice solution. We may want this to work on both R6 and pre-R6.
1788 // For now simply use the stack for RA. This should be OK since for the
1789 // vast majority of code a short PC-relative branch is sufficient.
1790 // TODO: can this be improved?
1791 Push(RA);
1792 Nal();
1793 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1794 Lui(AT, High16Bits(offset));
1795 Ori(AT, AT, Low16Bits(offset));
1796 Addu(AT, AT, RA);
1797 Lw(RA, SP, 0);
1798 Jr(AT);
1799 DecreaseFrameSize(kMipsWordSize);
1800 break;
1801 case Branch::kLongCondBranch:
1802 // The comment on case 'Branch::kLongUncondBranch' applies here as well.
1803 // Note: the opposite condition branch encodes 8 as the distance, which is equal to the
1804 // number of instructions skipped:
1805 // (PUSH(IncreaseFrameSize(ADDIU) + SW) + NAL + LUI + ORI + ADDU + LW + JR).
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001806 EmitBcondR2(Branch::OppositeCondition(condition), lhs, rhs, 8);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001807 Push(RA);
1808 Nal();
1809 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1810 Lui(AT, High16Bits(offset));
1811 Ori(AT, AT, Low16Bits(offset));
1812 Addu(AT, AT, RA);
1813 Lw(RA, SP, 0);
1814 Jr(AT);
1815 DecreaseFrameSize(kMipsWordSize);
1816 break;
1817 case Branch::kLongCall:
1818 Nal();
1819 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1820 Lui(AT, High16Bits(offset));
1821 Ori(AT, AT, Low16Bits(offset));
1822 Addu(lhs, AT, RA);
1823 Jalr(lhs);
1824 Nop();
1825 break;
1826
1827 // R6 short branches.
1828 case Branch::kR6UncondBranch:
1829 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1830 Bc(offset);
1831 break;
1832 case Branch::kR6CondBranch:
1833 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001834 EmitBcondR6(condition, lhs, rhs, offset);
1835 Nop(); // TODO: improve by filling the forbidden/delay slot.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001836 break;
1837 case Branch::kR6Call:
1838 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1839 Addiupc(lhs, offset);
1840 Jialc(lhs, 0);
1841 break;
1842
1843 // R6 long branches.
1844 case Branch::kR6LongUncondBranch:
1845 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1846 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1847 Auipc(AT, High16Bits(offset));
1848 Jic(AT, Low16Bits(offset));
1849 break;
1850 case Branch::kR6LongCondBranch:
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001851 EmitBcondR6(Branch::OppositeCondition(condition), lhs, rhs, 2);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001852 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1853 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1854 Auipc(AT, High16Bits(offset));
1855 Jic(AT, Low16Bits(offset));
1856 break;
1857 case Branch::kR6LongCall:
1858 offset += (offset & 0x8000) << 1; // Account for sign extension in addiu.
1859 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1860 Auipc(lhs, High16Bits(offset));
1861 Addiu(lhs, lhs, Low16Bits(offset));
1862 Jialc(lhs, 0);
1863 break;
1864 }
1865 CHECK_EQ(overwrite_location_, branch->GetEndLocation());
1866 CHECK_LT(branch->GetSize(), static_cast<uint32_t>(Branch::kMaxBranchSize));
1867}
1868
1869void MipsAssembler::B(MipsLabel* label) {
1870 Buncond(label);
1871}
1872
1873void MipsAssembler::Jalr(MipsLabel* label, Register indirect_reg) {
1874 Call(label, indirect_reg);
1875}
1876
1877void MipsAssembler::Beq(Register rs, Register rt, MipsLabel* label) {
1878 Bcond(label, kCondEQ, rs, rt);
1879}
1880
1881void MipsAssembler::Bne(Register rs, Register rt, MipsLabel* label) {
1882 Bcond(label, kCondNE, rs, rt);
1883}
1884
1885void MipsAssembler::Beqz(Register rt, MipsLabel* label) {
1886 Bcond(label, kCondEQZ, rt);
1887}
1888
1889void MipsAssembler::Bnez(Register rt, MipsLabel* label) {
1890 Bcond(label, kCondNEZ, rt);
1891}
1892
1893void MipsAssembler::Bltz(Register rt, MipsLabel* label) {
1894 Bcond(label, kCondLTZ, rt);
1895}
1896
1897void MipsAssembler::Bgez(Register rt, MipsLabel* label) {
1898 Bcond(label, kCondGEZ, rt);
1899}
1900
1901void MipsAssembler::Blez(Register rt, MipsLabel* label) {
1902 Bcond(label, kCondLEZ, rt);
1903}
1904
1905void MipsAssembler::Bgtz(Register rt, MipsLabel* label) {
1906 Bcond(label, kCondGTZ, rt);
1907}
1908
1909void MipsAssembler::Blt(Register rs, Register rt, MipsLabel* label) {
1910 if (IsR6()) {
1911 Bcond(label, kCondLT, rs, rt);
1912 } else if (!Branch::IsNop(kCondLT, rs, rt)) {
1913 // Synthesize the instruction (not available on R2).
1914 Slt(AT, rs, rt);
1915 Bnez(AT, label);
1916 }
1917}
1918
1919void MipsAssembler::Bge(Register rs, Register rt, MipsLabel* label) {
1920 if (IsR6()) {
1921 Bcond(label, kCondGE, rs, rt);
1922 } else if (Branch::IsUncond(kCondGE, rs, rt)) {
1923 B(label);
1924 } else {
1925 // Synthesize the instruction (not available on R2).
1926 Slt(AT, rs, rt);
1927 Beqz(AT, label);
1928 }
1929}
1930
1931void MipsAssembler::Bltu(Register rs, Register rt, MipsLabel* label) {
1932 if (IsR6()) {
1933 Bcond(label, kCondLTU, rs, rt);
1934 } else if (!Branch::IsNop(kCondLTU, rs, rt)) {
1935 // Synthesize the instruction (not available on R2).
1936 Sltu(AT, rs, rt);
1937 Bnez(AT, label);
1938 }
1939}
1940
1941void MipsAssembler::Bgeu(Register rs, Register rt, MipsLabel* label) {
1942 if (IsR6()) {
1943 Bcond(label, kCondGEU, rs, rt);
1944 } else if (Branch::IsUncond(kCondGEU, rs, rt)) {
1945 B(label);
1946 } else {
1947 // Synthesize the instruction (not available on R2).
1948 Sltu(AT, rs, rt);
1949 Beqz(AT, label);
jeffhao7fbee072012-08-24 17:56:54 -07001950 }
1951}
1952
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001953void MipsAssembler::Bc1f(int cc, MipsLabel* label) {
1954 CHECK(IsUint<3>(cc)) << cc;
1955 Bcond(label, kCondF, static_cast<Register>(cc), ZERO);
1956}
1957
1958void MipsAssembler::Bc1t(int cc, MipsLabel* label) {
1959 CHECK(IsUint<3>(cc)) << cc;
1960 Bcond(label, kCondT, static_cast<Register>(cc), ZERO);
1961}
1962
1963void MipsAssembler::Bc1eqz(FRegister ft, MipsLabel* label) {
1964 Bcond(label, kCondF, static_cast<Register>(ft), ZERO);
1965}
1966
1967void MipsAssembler::Bc1nez(FRegister ft, MipsLabel* label) {
1968 Bcond(label, kCondT, static_cast<Register>(ft), ZERO);
1969}
1970
jeffhao7fbee072012-08-24 17:56:54 -07001971void MipsAssembler::LoadFromOffset(LoadOperandType type, Register reg, Register base,
1972 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001973 // IsInt<16> must be passed a signed value.
1974 if (!IsInt<16>(offset) ||
1975 (type == kLoadDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
1976 LoadConst32(AT, offset);
1977 Addu(AT, AT, base);
1978 base = AT;
1979 offset = 0;
1980 }
1981
jeffhao7fbee072012-08-24 17:56:54 -07001982 switch (type) {
1983 case kLoadSignedByte:
1984 Lb(reg, base, offset);
1985 break;
1986 case kLoadUnsignedByte:
1987 Lbu(reg, base, offset);
1988 break;
1989 case kLoadSignedHalfword:
1990 Lh(reg, base, offset);
1991 break;
1992 case kLoadUnsignedHalfword:
1993 Lhu(reg, base, offset);
1994 break;
1995 case kLoadWord:
1996 Lw(reg, base, offset);
1997 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001998 case kLoadDoubleword:
1999 if (reg == base) {
2000 // This will clobber the base when loading the lower register. Since we have to load the
2001 // higher register as well, this will fail. Solution: reverse the order.
2002 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
2003 Lw(reg, base, offset);
2004 } else {
2005 Lw(reg, base, offset);
2006 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
2007 }
jeffhao7fbee072012-08-24 17:56:54 -07002008 break;
2009 default:
2010 LOG(FATAL) << "UNREACHABLE";
2011 }
2012}
2013
2014void MipsAssembler::LoadSFromOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002015 if (!IsInt<16>(offset)) {
2016 LoadConst32(AT, offset);
2017 Addu(AT, AT, base);
2018 base = AT;
2019 offset = 0;
2020 }
2021
jeffhao7fbee072012-08-24 17:56:54 -07002022 Lwc1(reg, base, offset);
2023}
2024
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002025void MipsAssembler::LoadDFromOffset(FRegister reg, Register base, int32_t offset) {
2026 // IsInt<16> must be passed a signed value.
2027 if (!IsInt<16>(offset) ||
2028 (!IsAligned<kMipsDoublewordSize>(offset) &&
2029 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2030 LoadConst32(AT, offset);
2031 Addu(AT, AT, base);
2032 base = AT;
2033 offset = 0;
2034 }
2035
2036 if (offset & 0x7) {
2037 if (Is32BitFPU()) {
2038 Lwc1(reg, base, offset);
2039 Lwc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
2040 } else {
2041 // 64-bit FPU.
2042 Lwc1(reg, base, offset);
2043 Lw(T8, base, offset + kMipsWordSize);
2044 Mthc1(T8, reg);
2045 }
2046 } else {
2047 Ldc1(reg, base, offset);
2048 }
2049}
2050
2051void MipsAssembler::EmitLoad(ManagedRegister m_dst, Register src_register, int32_t src_offset,
2052 size_t size) {
2053 MipsManagedRegister dst = m_dst.AsMips();
2054 if (dst.IsNoRegister()) {
2055 CHECK_EQ(0u, size) << dst;
2056 } else if (dst.IsCoreRegister()) {
2057 CHECK_EQ(kMipsWordSize, size) << dst;
2058 LoadFromOffset(kLoadWord, dst.AsCoreRegister(), src_register, src_offset);
2059 } else if (dst.IsRegisterPair()) {
2060 CHECK_EQ(kMipsDoublewordSize, size) << dst;
2061 LoadFromOffset(kLoadDoubleword, dst.AsRegisterPairLow(), src_register, src_offset);
2062 } else if (dst.IsFRegister()) {
2063 if (size == kMipsWordSize) {
2064 LoadSFromOffset(dst.AsFRegister(), src_register, src_offset);
2065 } else {
2066 CHECK_EQ(kMipsDoublewordSize, size) << dst;
2067 LoadDFromOffset(dst.AsFRegister(), src_register, src_offset);
2068 }
2069 }
jeffhao7fbee072012-08-24 17:56:54 -07002070}
2071
2072void MipsAssembler::StoreToOffset(StoreOperandType type, Register reg, Register base,
2073 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002074 // IsInt<16> must be passed a signed value.
2075 if (!IsInt<16>(offset) ||
2076 (type == kStoreDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2077 LoadConst32(AT, offset);
2078 Addu(AT, AT, base);
2079 base = AT;
2080 offset = 0;
2081 }
2082
jeffhao7fbee072012-08-24 17:56:54 -07002083 switch (type) {
2084 case kStoreByte:
2085 Sb(reg, base, offset);
2086 break;
2087 case kStoreHalfword:
2088 Sh(reg, base, offset);
2089 break;
2090 case kStoreWord:
2091 Sw(reg, base, offset);
2092 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002093 case kStoreDoubleword:
2094 CHECK_NE(reg, base);
2095 CHECK_NE(static_cast<Register>(reg + 1), base);
2096 Sw(reg, base, offset);
2097 Sw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002098 break;
2099 default:
2100 LOG(FATAL) << "UNREACHABLE";
2101 }
2102}
2103
Goran Jakovljevicff734982015-08-24 12:58:55 +00002104void MipsAssembler::StoreSToOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002105 if (!IsInt<16>(offset)) {
2106 LoadConst32(AT, offset);
2107 Addu(AT, AT, base);
2108 base = AT;
2109 offset = 0;
2110 }
2111
jeffhao7fbee072012-08-24 17:56:54 -07002112 Swc1(reg, base, offset);
2113}
2114
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002115void MipsAssembler::StoreDToOffset(FRegister reg, Register base, int32_t offset) {
2116 // IsInt<16> must be passed a signed value.
2117 if (!IsInt<16>(offset) ||
2118 (!IsAligned<kMipsDoublewordSize>(offset) &&
2119 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2120 LoadConst32(AT, offset);
2121 Addu(AT, AT, base);
2122 base = AT;
2123 offset = 0;
2124 }
2125
2126 if (offset & 0x7) {
2127 if (Is32BitFPU()) {
2128 Swc1(reg, base, offset);
2129 Swc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
2130 } else {
2131 // 64-bit FPU.
2132 Mfhc1(T8, reg);
2133 Swc1(reg, base, offset);
2134 Sw(T8, base, offset + kMipsWordSize);
2135 }
2136 } else {
2137 Sdc1(reg, base, offset);
2138 }
jeffhao7fbee072012-08-24 17:56:54 -07002139}
2140
David Srbeckydd973932015-04-07 20:29:48 +01002141static dwarf::Reg DWARFReg(Register reg) {
2142 return dwarf::Reg::MipsCore(static_cast<int>(reg));
2143}
2144
Ian Rogers790a6b72014-04-01 10:36:00 -07002145constexpr size_t kFramePointerSize = 4;
2146
jeffhao7fbee072012-08-24 17:56:54 -07002147void MipsAssembler::BuildFrame(size_t frame_size, ManagedRegister method_reg,
2148 const std::vector<ManagedRegister>& callee_save_regs,
Dmitry Petrochenkofca82202014-03-21 11:21:37 +07002149 const ManagedRegisterEntrySpills& entry_spills) {
jeffhao7fbee072012-08-24 17:56:54 -07002150 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002151 DCHECK(!overwriting_);
jeffhao7fbee072012-08-24 17:56:54 -07002152
2153 // Increase frame to required size.
2154 IncreaseFrameSize(frame_size);
2155
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002156 // Push callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07002157 int stack_offset = frame_size - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002158 StoreToOffset(kStoreWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002159 cfi_.RelOffset(DWARFReg(RA), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07002160 for (int i = callee_save_regs.size() - 1; i >= 0; --i) {
Ian Rogers790a6b72014-04-01 10:36:00 -07002161 stack_offset -= kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002162 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
2163 StoreToOffset(kStoreWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002164 cfi_.RelOffset(DWARFReg(reg), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07002165 }
2166
2167 // Write out Method*.
2168 StoreToOffset(kStoreWord, method_reg.AsMips().AsCoreRegister(), SP, 0);
2169
2170 // Write out entry spills.
Goran Jakovljevicff734982015-08-24 12:58:55 +00002171 int32_t offset = frame_size + kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002172 for (size_t i = 0; i < entry_spills.size(); ++i) {
Goran Jakovljevicff734982015-08-24 12:58:55 +00002173 MipsManagedRegister reg = entry_spills.at(i).AsMips();
2174 if (reg.IsNoRegister()) {
2175 ManagedRegisterSpill spill = entry_spills.at(i);
2176 offset += spill.getSize();
2177 } else if (reg.IsCoreRegister()) {
2178 StoreToOffset(kStoreWord, reg.AsCoreRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002179 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002180 } else if (reg.IsFRegister()) {
2181 StoreSToOffset(reg.AsFRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002182 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002183 } else if (reg.IsDRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002184 StoreDToOffset(reg.AsOverlappingDRegisterLow(), SP, offset);
2185 offset += kMipsDoublewordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002186 }
jeffhao7fbee072012-08-24 17:56:54 -07002187 }
2188}
2189
2190void MipsAssembler::RemoveFrame(size_t frame_size,
2191 const std::vector<ManagedRegister>& callee_save_regs) {
2192 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002193 DCHECK(!overwriting_);
David Srbeckydd973932015-04-07 20:29:48 +01002194 cfi_.RememberState();
jeffhao7fbee072012-08-24 17:56:54 -07002195
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002196 // Pop callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07002197 int stack_offset = frame_size - (callee_save_regs.size() * kFramePointerSize) - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002198 for (size_t i = 0; i < callee_save_regs.size(); ++i) {
2199 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
2200 LoadFromOffset(kLoadWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002201 cfi_.Restore(DWARFReg(reg));
Ian Rogers790a6b72014-04-01 10:36:00 -07002202 stack_offset += kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002203 }
2204 LoadFromOffset(kLoadWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002205 cfi_.Restore(DWARFReg(RA));
jeffhao7fbee072012-08-24 17:56:54 -07002206
2207 // Decrease frame to required size.
2208 DecreaseFrameSize(frame_size);
jeffhao07030602012-09-26 14:33:14 -07002209
2210 // Then jump to the return address.
2211 Jr(RA);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002212 Nop();
David Srbeckydd973932015-04-07 20:29:48 +01002213
2214 // The CFI should be restored for any code that follows the exit block.
2215 cfi_.RestoreState();
2216 cfi_.DefCFAOffset(frame_size);
jeffhao7fbee072012-08-24 17:56:54 -07002217}
2218
2219void MipsAssembler::IncreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002220 CHECK_ALIGNED(adjust, kFramePointerSize);
2221 Addiu32(SP, SP, -adjust);
David Srbeckydd973932015-04-07 20:29:48 +01002222 cfi_.AdjustCFAOffset(adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002223 if (overwriting_) {
2224 cfi_.OverrideDelayedPC(overwrite_location_);
2225 }
jeffhao7fbee072012-08-24 17:56:54 -07002226}
2227
2228void MipsAssembler::DecreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002229 CHECK_ALIGNED(adjust, kFramePointerSize);
2230 Addiu32(SP, SP, adjust);
David Srbeckydd973932015-04-07 20:29:48 +01002231 cfi_.AdjustCFAOffset(-adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002232 if (overwriting_) {
2233 cfi_.OverrideDelayedPC(overwrite_location_);
2234 }
jeffhao7fbee072012-08-24 17:56:54 -07002235}
2236
2237void MipsAssembler::Store(FrameOffset dest, ManagedRegister msrc, size_t size) {
2238 MipsManagedRegister src = msrc.AsMips();
2239 if (src.IsNoRegister()) {
2240 CHECK_EQ(0u, size);
2241 } else if (src.IsCoreRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002242 CHECK_EQ(kMipsWordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07002243 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2244 } else if (src.IsRegisterPair()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002245 CHECK_EQ(kMipsDoublewordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07002246 StoreToOffset(kStoreWord, src.AsRegisterPairLow(), SP, dest.Int32Value());
2247 StoreToOffset(kStoreWord, src.AsRegisterPairHigh(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002248 SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002249 } else if (src.IsFRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002250 if (size == kMipsWordSize) {
2251 StoreSToOffset(src.AsFRegister(), SP, dest.Int32Value());
2252 } else {
2253 CHECK_EQ(kMipsDoublewordSize, size);
2254 StoreDToOffset(src.AsFRegister(), SP, dest.Int32Value());
2255 }
jeffhao7fbee072012-08-24 17:56:54 -07002256 }
2257}
2258
2259void MipsAssembler::StoreRef(FrameOffset dest, ManagedRegister msrc) {
2260 MipsManagedRegister src = msrc.AsMips();
2261 CHECK(src.IsCoreRegister());
2262 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2263}
2264
2265void MipsAssembler::StoreRawPtr(FrameOffset dest, ManagedRegister msrc) {
2266 MipsManagedRegister src = msrc.AsMips();
2267 CHECK(src.IsCoreRegister());
2268 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2269}
2270
2271void MipsAssembler::StoreImmediateToFrame(FrameOffset dest, uint32_t imm,
2272 ManagedRegister mscratch) {
2273 MipsManagedRegister scratch = mscratch.AsMips();
2274 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002275 LoadConst32(scratch.AsCoreRegister(), imm);
jeffhao7fbee072012-08-24 17:56:54 -07002276 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2277}
2278
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002279void MipsAssembler::StoreImmediateToThread32(ThreadOffset<kMipsWordSize> dest, uint32_t imm,
jeffhao7fbee072012-08-24 17:56:54 -07002280 ManagedRegister mscratch) {
2281 MipsManagedRegister scratch = mscratch.AsMips();
2282 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002283 // Is this function even referenced anywhere else in the code?
2284 LoadConst32(scratch.AsCoreRegister(), imm);
2285 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), S1, dest.Int32Value());
2286}
2287
2288void MipsAssembler::StoreStackOffsetToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2289 FrameOffset fr_offs,
2290 ManagedRegister mscratch) {
2291 MipsManagedRegister scratch = mscratch.AsMips();
2292 CHECK(scratch.IsCoreRegister()) << scratch;
2293 Addiu32(scratch.AsCoreRegister(), SP, fr_offs.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002294 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2295 S1, thr_offs.Int32Value());
2296}
2297
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002298void MipsAssembler::StoreStackPointerToThread32(ThreadOffset<kMipsWordSize> thr_offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002299 StoreToOffset(kStoreWord, SP, S1, thr_offs.Int32Value());
2300}
2301
2302void MipsAssembler::StoreSpanning(FrameOffset dest, ManagedRegister msrc,
2303 FrameOffset in_off, ManagedRegister mscratch) {
2304 MipsManagedRegister src = msrc.AsMips();
2305 MipsManagedRegister scratch = mscratch.AsMips();
2306 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2307 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, in_off.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002308 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002309}
2310
2311void MipsAssembler::Load(ManagedRegister mdest, FrameOffset src, size_t size) {
2312 return EmitLoad(mdest, SP, src.Int32Value(), size);
2313}
2314
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002315void MipsAssembler::LoadFromThread32(ManagedRegister mdest,
2316 ThreadOffset<kMipsWordSize> src, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002317 return EmitLoad(mdest, S1, src.Int32Value(), size);
2318}
2319
2320void MipsAssembler::LoadRef(ManagedRegister mdest, FrameOffset src) {
2321 MipsManagedRegister dest = mdest.AsMips();
2322 CHECK(dest.IsCoreRegister());
2323 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), SP, src.Int32Value());
2324}
2325
Mathieu Chartiere401d142015-04-22 13:56:20 -07002326void MipsAssembler::LoadRef(ManagedRegister mdest, ManagedRegister base, MemberOffset offs,
Roland Levillain4d027112015-07-01 15:41:14 +01002327 bool unpoison_reference) {
jeffhao7fbee072012-08-24 17:56:54 -07002328 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002329 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002330 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2331 base.AsMips().AsCoreRegister(), offs.Int32Value());
Roland Levillain4d027112015-07-01 15:41:14 +01002332 if (kPoisonHeapReferences && unpoison_reference) {
Hiroshi Yamauchie63a7452014-02-27 14:44:36 -08002333 Subu(dest.AsCoreRegister(), ZERO, dest.AsCoreRegister());
2334 }
jeffhao7fbee072012-08-24 17:56:54 -07002335}
2336
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002337void MipsAssembler::LoadRawPtr(ManagedRegister mdest, ManagedRegister base, Offset offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002338 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002339 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002340 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2341 base.AsMips().AsCoreRegister(), offs.Int32Value());
2342}
2343
Ian Rogersdd7624d2014-03-14 17:43:00 -07002344void MipsAssembler::LoadRawPtrFromThread32(ManagedRegister mdest,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002345 ThreadOffset<kMipsWordSize> offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002346 MipsManagedRegister dest = mdest.AsMips();
2347 CHECK(dest.IsCoreRegister());
2348 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), S1, offs.Int32Value());
2349}
2350
2351void MipsAssembler::SignExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2352 UNIMPLEMENTED(FATAL) << "no sign extension necessary for mips";
2353}
2354
2355void MipsAssembler::ZeroExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2356 UNIMPLEMENTED(FATAL) << "no zero extension necessary for mips";
2357}
2358
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002359void MipsAssembler::Move(ManagedRegister mdest, ManagedRegister msrc, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002360 MipsManagedRegister dest = mdest.AsMips();
2361 MipsManagedRegister src = msrc.AsMips();
2362 if (!dest.Equals(src)) {
2363 if (dest.IsCoreRegister()) {
2364 CHECK(src.IsCoreRegister()) << src;
2365 Move(dest.AsCoreRegister(), src.AsCoreRegister());
2366 } else if (dest.IsFRegister()) {
2367 CHECK(src.IsFRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002368 if (size == kMipsWordSize) {
2369 MovS(dest.AsFRegister(), src.AsFRegister());
2370 } else {
2371 CHECK_EQ(kMipsDoublewordSize, size);
2372 MovD(dest.AsFRegister(), src.AsFRegister());
2373 }
jeffhao7fbee072012-08-24 17:56:54 -07002374 } else if (dest.IsDRegister()) {
2375 CHECK(src.IsDRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002376 MovD(dest.AsOverlappingDRegisterLow(), src.AsOverlappingDRegisterLow());
jeffhao7fbee072012-08-24 17:56:54 -07002377 } else {
2378 CHECK(dest.IsRegisterPair()) << dest;
2379 CHECK(src.IsRegisterPair()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002380 // Ensure that the first move doesn't clobber the input of the second.
jeffhao7fbee072012-08-24 17:56:54 -07002381 if (src.AsRegisterPairHigh() != dest.AsRegisterPairLow()) {
2382 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2383 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2384 } else {
2385 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2386 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2387 }
2388 }
2389 }
2390}
2391
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002392void MipsAssembler::CopyRef(FrameOffset dest, FrameOffset src, ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002393 MipsManagedRegister scratch = mscratch.AsMips();
2394 CHECK(scratch.IsCoreRegister()) << scratch;
2395 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2396 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2397}
2398
Ian Rogersdd7624d2014-03-14 17:43:00 -07002399void MipsAssembler::CopyRawPtrFromThread32(FrameOffset fr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002400 ThreadOffset<kMipsWordSize> thr_offs,
2401 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002402 MipsManagedRegister scratch = mscratch.AsMips();
2403 CHECK(scratch.IsCoreRegister()) << scratch;
2404 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2405 S1, thr_offs.Int32Value());
2406 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2407 SP, fr_offs.Int32Value());
2408}
2409
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002410void MipsAssembler::CopyRawPtrToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2411 FrameOffset fr_offs,
2412 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002413 MipsManagedRegister scratch = mscratch.AsMips();
2414 CHECK(scratch.IsCoreRegister()) << scratch;
2415 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2416 SP, fr_offs.Int32Value());
2417 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2418 S1, thr_offs.Int32Value());
2419}
2420
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002421void MipsAssembler::Copy(FrameOffset dest, FrameOffset src, ManagedRegister mscratch, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002422 MipsManagedRegister scratch = mscratch.AsMips();
2423 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002424 CHECK(size == kMipsWordSize || size == kMipsDoublewordSize) << size;
2425 if (size == kMipsWordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002426 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2427 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002428 } else if (size == kMipsDoublewordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002429 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2430 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002431 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value() + kMipsWordSize);
2432 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002433 }
2434}
2435
2436void MipsAssembler::Copy(FrameOffset dest, ManagedRegister src_base, Offset src_offset,
2437 ManagedRegister mscratch, size_t size) {
2438 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002439 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002440 LoadFromOffset(kLoadWord, scratch, src_base.AsMips().AsCoreRegister(), src_offset.Int32Value());
2441 StoreToOffset(kStoreWord, scratch, SP, dest.Int32Value());
2442}
2443
2444void MipsAssembler::Copy(ManagedRegister dest_base, Offset dest_offset, FrameOffset src,
2445 ManagedRegister mscratch, size_t size) {
2446 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002447 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002448 LoadFromOffset(kLoadWord, scratch, SP, src.Int32Value());
2449 StoreToOffset(kStoreWord, scratch, dest_base.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2450}
2451
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002452void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2453 FrameOffset src_base ATTRIBUTE_UNUSED,
2454 Offset src_offset ATTRIBUTE_UNUSED,
2455 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2456 size_t size ATTRIBUTE_UNUSED) {
2457 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002458}
2459
2460void MipsAssembler::Copy(ManagedRegister dest, Offset dest_offset,
2461 ManagedRegister src, Offset src_offset,
2462 ManagedRegister mscratch, size_t size) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002463 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002464 Register scratch = mscratch.AsMips().AsCoreRegister();
2465 LoadFromOffset(kLoadWord, scratch, src.AsMips().AsCoreRegister(), src_offset.Int32Value());
2466 StoreToOffset(kStoreWord, scratch, dest.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2467}
2468
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002469void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2470 Offset dest_offset ATTRIBUTE_UNUSED,
2471 FrameOffset src ATTRIBUTE_UNUSED,
2472 Offset src_offset ATTRIBUTE_UNUSED,
2473 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2474 size_t size ATTRIBUTE_UNUSED) {
2475 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002476}
2477
2478void MipsAssembler::MemoryBarrier(ManagedRegister) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002479 // TODO: sync?
2480 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002481}
2482
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002483void MipsAssembler::CreateHandleScopeEntry(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002484 FrameOffset handle_scope_offset,
2485 ManagedRegister min_reg,
2486 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002487 MipsManagedRegister out_reg = mout_reg.AsMips();
2488 MipsManagedRegister in_reg = min_reg.AsMips();
2489 CHECK(in_reg.IsNoRegister() || in_reg.IsCoreRegister()) << in_reg;
2490 CHECK(out_reg.IsCoreRegister()) << out_reg;
2491 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002492 MipsLabel null_arg;
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002493 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2494 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002495 // E.g. out_reg = (handle == 0) ? 0 : (SP+handle_offset).
jeffhao7fbee072012-08-24 17:56:54 -07002496 if (in_reg.IsNoRegister()) {
2497 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002498 SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002499 in_reg = out_reg;
2500 }
2501 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002502 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002503 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002504 Beqz(in_reg.AsCoreRegister(), &null_arg);
2505 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2506 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002507 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002508 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002509 }
2510}
2511
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002512void MipsAssembler::CreateHandleScopeEntry(FrameOffset out_off,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002513 FrameOffset handle_scope_offset,
2514 ManagedRegister mscratch,
2515 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002516 MipsManagedRegister scratch = mscratch.AsMips();
2517 CHECK(scratch.IsCoreRegister()) << scratch;
2518 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002519 MipsLabel null_arg;
2520 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002521 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2522 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002523 // E.g. scratch = (scratch == 0) ? 0 : (SP+handle_scope_offset).
2524 Beqz(scratch.AsCoreRegister(), &null_arg);
2525 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2526 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002527 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002528 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002529 }
2530 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, out_off.Int32Value());
2531}
2532
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002533// Given a handle scope entry, load the associated reference.
2534void MipsAssembler::LoadReferenceFromHandleScope(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002535 ManagedRegister min_reg) {
jeffhao7fbee072012-08-24 17:56:54 -07002536 MipsManagedRegister out_reg = mout_reg.AsMips();
2537 MipsManagedRegister in_reg = min_reg.AsMips();
2538 CHECK(out_reg.IsCoreRegister()) << out_reg;
2539 CHECK(in_reg.IsCoreRegister()) << in_reg;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002540 MipsLabel null_arg;
jeffhao7fbee072012-08-24 17:56:54 -07002541 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002542 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002543 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002544 Beqz(in_reg.AsCoreRegister(), &null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002545 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
2546 in_reg.AsCoreRegister(), 0);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002547 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002548}
2549
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002550void MipsAssembler::VerifyObject(ManagedRegister src ATTRIBUTE_UNUSED,
2551 bool could_be_null ATTRIBUTE_UNUSED) {
2552 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002553}
2554
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002555void MipsAssembler::VerifyObject(FrameOffset src ATTRIBUTE_UNUSED,
2556 bool could_be_null ATTRIBUTE_UNUSED) {
2557 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002558}
2559
2560void MipsAssembler::Call(ManagedRegister mbase, Offset offset, ManagedRegister mscratch) {
2561 MipsManagedRegister base = mbase.AsMips();
2562 MipsManagedRegister scratch = mscratch.AsMips();
2563 CHECK(base.IsCoreRegister()) << base;
2564 CHECK(scratch.IsCoreRegister()) << scratch;
2565 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2566 base.AsCoreRegister(), offset.Int32Value());
2567 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002568 Nop();
2569 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002570}
2571
2572void MipsAssembler::Call(FrameOffset base, Offset offset, ManagedRegister mscratch) {
2573 MipsManagedRegister scratch = mscratch.AsMips();
2574 CHECK(scratch.IsCoreRegister()) << scratch;
2575 // Call *(*(SP + base) + offset)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002576 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, base.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002577 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2578 scratch.AsCoreRegister(), offset.Int32Value());
2579 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002580 Nop();
2581 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002582}
2583
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002584void MipsAssembler::CallFromThread32(ThreadOffset<kMipsWordSize> offset ATTRIBUTE_UNUSED,
2585 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
Ian Rogers468532e2013-08-05 10:56:33 -07002586 UNIMPLEMENTED(FATAL) << "no mips implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002587}
2588
2589void MipsAssembler::GetCurrentThread(ManagedRegister tr) {
2590 Move(tr.AsMips().AsCoreRegister(), S1);
2591}
2592
2593void MipsAssembler::GetCurrentThread(FrameOffset offset,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002594 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
jeffhao7fbee072012-08-24 17:56:54 -07002595 StoreToOffset(kStoreWord, S1, SP, offset.Int32Value());
2596}
2597
jeffhao7fbee072012-08-24 17:56:54 -07002598void MipsAssembler::ExceptionPoll(ManagedRegister mscratch, size_t stack_adjust) {
2599 MipsManagedRegister scratch = mscratch.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002600 exception_blocks_.emplace_back(scratch, stack_adjust);
jeffhao7fbee072012-08-24 17:56:54 -07002601 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002602 S1, Thread::ExceptionOffset<kMipsWordSize>().Int32Value());
2603 // TODO: on MIPS32R6 prefer Bnezc(scratch.AsCoreRegister(), slow.Entry());
2604 // as the NAL instruction (occurring in long R2 branches) may become deprecated.
2605 // For now use common for R2 and R6 instructions as this code must execute on both.
2606 Bnez(scratch.AsCoreRegister(), exception_blocks_.back().Entry());
jeffhao7fbee072012-08-24 17:56:54 -07002607}
2608
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002609void MipsAssembler::EmitExceptionPoll(MipsExceptionSlowPath* exception) {
2610 Bind(exception->Entry());
2611 if (exception->stack_adjust_ != 0) { // Fix up the frame.
2612 DecreaseFrameSize(exception->stack_adjust_);
jeffhao7fbee072012-08-24 17:56:54 -07002613 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002614 // Pass exception object as argument.
2615 // Don't care about preserving A0 as this call won't return.
2616 CheckEntrypointTypes<kQuickDeliverException, void, mirror::Object*>();
2617 Move(A0, exception->scratch_.AsCoreRegister());
2618 // Set up call to Thread::Current()->pDeliverException.
2619 LoadFromOffset(kLoadWord, T9, S1,
2620 QUICK_ENTRYPOINT_OFFSET(kMipsWordSize, pDeliverException).Int32Value());
2621 Jr(T9);
2622 Nop();
2623
2624 // Call never returns.
2625 Break();
jeffhao7fbee072012-08-24 17:56:54 -07002626}
2627
2628} // namespace mips
2629} // namespace art