blob: ac9c097892af636bf1d5e8d44a00164573627b4e [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
Alexey Frunze5c7aed32015-11-25 19:41:54 -0800403void MipsAssembler::Ext(Register rd, Register rt, int pos, int size) {
404 CHECK(IsUint<5>(pos)) << pos;
405 CHECK(0 < size && size <= 32) << size;
406 CHECK(0 < pos + size && pos + size <= 32) << pos << " + " << size;
407 EmitR(0x1f, rt, rd, static_cast<Register>(size - 1), pos, 0x00);
408}
409
410void MipsAssembler::Ins(Register rd, Register rt, int pos, int size) {
411 CHECK(IsUint<5>(pos)) << pos;
412 CHECK(0 < size && size <= 32) << size;
413 CHECK(0 < pos + size && pos + size <= 32) << pos << " + " << size;
414 EmitR(0x1f, rt, rd, static_cast<Register>(pos + size - 1), pos, 0x04);
415}
416
jeffhao7fbee072012-08-24 17:56:54 -0700417void MipsAssembler::Lb(Register rt, Register rs, uint16_t imm16) {
418 EmitI(0x20, rs, rt, imm16);
419}
420
421void MipsAssembler::Lh(Register rt, Register rs, uint16_t imm16) {
422 EmitI(0x21, rs, rt, imm16);
423}
424
425void MipsAssembler::Lw(Register rt, Register rs, uint16_t imm16) {
426 EmitI(0x23, rs, rt, imm16);
427}
428
429void MipsAssembler::Lbu(Register rt, Register rs, uint16_t imm16) {
430 EmitI(0x24, rs, rt, imm16);
431}
432
433void MipsAssembler::Lhu(Register rt, Register rs, uint16_t imm16) {
434 EmitI(0x25, rs, rt, imm16);
435}
436
437void MipsAssembler::Lui(Register rt, uint16_t imm16) {
438 EmitI(0xf, static_cast<Register>(0), rt, imm16);
439}
440
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200441void MipsAssembler::Sync(uint32_t stype) {
442 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0),
443 stype & 0x1f, 0xf);
444}
445
jeffhao7fbee072012-08-24 17:56:54 -0700446void MipsAssembler::Mfhi(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200447 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700448 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x10);
449}
450
451void MipsAssembler::Mflo(Register rd) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200452 CHECK(!IsR6());
jeffhao7fbee072012-08-24 17:56:54 -0700453 EmitR(0, static_cast<Register>(0), static_cast<Register>(0), rd, 0, 0x12);
454}
455
456void MipsAssembler::Sb(Register rt, Register rs, uint16_t imm16) {
457 EmitI(0x28, rs, rt, imm16);
458}
459
460void MipsAssembler::Sh(Register rt, Register rs, uint16_t imm16) {
461 EmitI(0x29, rs, rt, imm16);
462}
463
464void MipsAssembler::Sw(Register rt, Register rs, uint16_t imm16) {
465 EmitI(0x2b, rs, rt, imm16);
466}
467
468void MipsAssembler::Slt(Register rd, Register rs, Register rt) {
469 EmitR(0, rs, rt, rd, 0, 0x2a);
470}
471
472void MipsAssembler::Sltu(Register rd, Register rs, Register rt) {
473 EmitR(0, rs, rt, rd, 0, 0x2b);
474}
475
476void MipsAssembler::Slti(Register rt, Register rs, uint16_t imm16) {
477 EmitI(0xa, rs, rt, imm16);
478}
479
480void MipsAssembler::Sltiu(Register rt, Register rs, uint16_t imm16) {
481 EmitI(0xb, rs, rt, imm16);
482}
483
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200484void MipsAssembler::B(uint16_t imm16) {
485 EmitI(0x4, static_cast<Register>(0), static_cast<Register>(0), imm16);
486}
487
488void MipsAssembler::Beq(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700489 EmitI(0x4, rs, rt, imm16);
490}
491
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200492void MipsAssembler::Bne(Register rs, Register rt, uint16_t imm16) {
jeffhao7fbee072012-08-24 17:56:54 -0700493 EmitI(0x5, rs, rt, imm16);
494}
495
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200496void MipsAssembler::Beqz(Register rt, uint16_t imm16) {
497 Beq(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700498}
499
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200500void MipsAssembler::Bnez(Register rt, uint16_t imm16) {
501 Bne(ZERO, rt, imm16);
jeffhao7fbee072012-08-24 17:56:54 -0700502}
503
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200504void MipsAssembler::Bltz(Register rt, uint16_t imm16) {
505 EmitI(0x1, rt, static_cast<Register>(0), imm16);
506}
507
508void MipsAssembler::Bgez(Register rt, uint16_t imm16) {
509 EmitI(0x1, rt, static_cast<Register>(0x1), imm16);
510}
511
512void MipsAssembler::Blez(Register rt, uint16_t imm16) {
513 EmitI(0x6, rt, static_cast<Register>(0), imm16);
514}
515
516void MipsAssembler::Bgtz(Register rt, uint16_t imm16) {
517 EmitI(0x7, rt, static_cast<Register>(0), imm16);
518}
519
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800520void MipsAssembler::Bc1f(int cc, uint16_t imm16) {
521 CHECK(!IsR6());
522 CHECK(IsUint<3>(cc)) << cc;
523 EmitI(0x11, static_cast<Register>(0x8), static_cast<Register>(cc << 2), imm16);
524}
525
526void MipsAssembler::Bc1t(int cc, uint16_t imm16) {
527 CHECK(!IsR6());
528 CHECK(IsUint<3>(cc)) << cc;
529 EmitI(0x11, static_cast<Register>(0x8), static_cast<Register>((cc << 2) | 1), imm16);
530}
531
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200532void MipsAssembler::J(uint32_t addr26) {
533 EmitI26(0x2, addr26);
534}
535
536void MipsAssembler::Jal(uint32_t addr26) {
537 EmitI26(0x3, addr26);
538}
539
540void MipsAssembler::Jalr(Register rd, Register rs) {
541 EmitR(0, rs, static_cast<Register>(0), rd, 0, 0x09);
jeffhao7fbee072012-08-24 17:56:54 -0700542}
543
544void MipsAssembler::Jalr(Register rs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200545 Jalr(RA, rs);
546}
547
548void MipsAssembler::Jr(Register rs) {
549 Jalr(ZERO, rs);
550}
551
552void MipsAssembler::Nal() {
553 EmitI(0x1, static_cast<Register>(0), static_cast<Register>(0x10), 0);
554}
555
556void MipsAssembler::Auipc(Register rs, uint16_t imm16) {
557 CHECK(IsR6());
558 EmitI(0x3B, rs, static_cast<Register>(0x1E), imm16);
559}
560
561void MipsAssembler::Addiupc(Register rs, uint32_t imm19) {
562 CHECK(IsR6());
563 CHECK(IsUint<19>(imm19)) << imm19;
564 EmitI21(0x3B, rs, imm19);
565}
566
567void MipsAssembler::Bc(uint32_t imm26) {
568 CHECK(IsR6());
569 EmitI26(0x32, imm26);
570}
571
572void MipsAssembler::Jic(Register rt, uint16_t imm16) {
573 CHECK(IsR6());
574 EmitI(0x36, static_cast<Register>(0), rt, imm16);
575}
576
577void MipsAssembler::Jialc(Register rt, uint16_t imm16) {
578 CHECK(IsR6());
579 EmitI(0x3E, static_cast<Register>(0), rt, imm16);
580}
581
582void MipsAssembler::Bltc(Register rs, Register rt, uint16_t imm16) {
583 CHECK(IsR6());
584 CHECK_NE(rs, ZERO);
585 CHECK_NE(rt, ZERO);
586 CHECK_NE(rs, rt);
587 EmitI(0x17, rs, rt, imm16);
588}
589
590void MipsAssembler::Bltzc(Register rt, uint16_t imm16) {
591 CHECK(IsR6());
592 CHECK_NE(rt, ZERO);
593 EmitI(0x17, rt, rt, imm16);
594}
595
596void MipsAssembler::Bgtzc(Register rt, uint16_t imm16) {
597 CHECK(IsR6());
598 CHECK_NE(rt, ZERO);
599 EmitI(0x17, static_cast<Register>(0), rt, imm16);
600}
601
602void MipsAssembler::Bgec(Register rs, Register rt, uint16_t imm16) {
603 CHECK(IsR6());
604 CHECK_NE(rs, ZERO);
605 CHECK_NE(rt, ZERO);
606 CHECK_NE(rs, rt);
607 EmitI(0x16, rs, rt, imm16);
608}
609
610void MipsAssembler::Bgezc(Register rt, uint16_t imm16) {
611 CHECK(IsR6());
612 CHECK_NE(rt, ZERO);
613 EmitI(0x16, rt, rt, imm16);
614}
615
616void MipsAssembler::Blezc(Register rt, uint16_t imm16) {
617 CHECK(IsR6());
618 CHECK_NE(rt, ZERO);
619 EmitI(0x16, static_cast<Register>(0), rt, imm16);
620}
621
622void MipsAssembler::Bltuc(Register rs, Register rt, uint16_t imm16) {
623 CHECK(IsR6());
624 CHECK_NE(rs, ZERO);
625 CHECK_NE(rt, ZERO);
626 CHECK_NE(rs, rt);
627 EmitI(0x7, rs, rt, imm16);
628}
629
630void MipsAssembler::Bgeuc(Register rs, Register rt, uint16_t imm16) {
631 CHECK(IsR6());
632 CHECK_NE(rs, ZERO);
633 CHECK_NE(rt, ZERO);
634 CHECK_NE(rs, rt);
635 EmitI(0x6, rs, rt, imm16);
636}
637
638void MipsAssembler::Beqc(Register rs, Register rt, uint16_t imm16) {
639 CHECK(IsR6());
640 CHECK_NE(rs, ZERO);
641 CHECK_NE(rt, ZERO);
642 CHECK_NE(rs, rt);
643 EmitI(0x8, std::min(rs, rt), std::max(rs, rt), imm16);
644}
645
646void MipsAssembler::Bnec(Register rs, Register rt, uint16_t imm16) {
647 CHECK(IsR6());
648 CHECK_NE(rs, ZERO);
649 CHECK_NE(rt, ZERO);
650 CHECK_NE(rs, rt);
651 EmitI(0x18, std::min(rs, rt), std::max(rs, rt), imm16);
652}
653
654void MipsAssembler::Beqzc(Register rs, uint32_t imm21) {
655 CHECK(IsR6());
656 CHECK_NE(rs, ZERO);
657 EmitI21(0x36, rs, imm21);
658}
659
660void MipsAssembler::Bnezc(Register rs, uint32_t imm21) {
661 CHECK(IsR6());
662 CHECK_NE(rs, ZERO);
663 EmitI21(0x3E, rs, imm21);
664}
665
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800666void MipsAssembler::Bc1eqz(FRegister ft, uint16_t imm16) {
667 CHECK(IsR6());
668 EmitFI(0x11, 0x9, ft, imm16);
669}
670
671void MipsAssembler::Bc1nez(FRegister ft, uint16_t imm16) {
672 CHECK(IsR6());
673 EmitFI(0x11, 0xD, ft, imm16);
674}
675
676void MipsAssembler::EmitBcondR2(BranchCondition cond, Register rs, Register rt, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200677 switch (cond) {
678 case kCondLTZ:
679 CHECK_EQ(rt, ZERO);
680 Bltz(rs, imm16);
681 break;
682 case kCondGEZ:
683 CHECK_EQ(rt, ZERO);
684 Bgez(rs, imm16);
685 break;
686 case kCondLEZ:
687 CHECK_EQ(rt, ZERO);
688 Blez(rs, imm16);
689 break;
690 case kCondGTZ:
691 CHECK_EQ(rt, ZERO);
692 Bgtz(rs, imm16);
693 break;
694 case kCondEQ:
695 Beq(rs, rt, imm16);
696 break;
697 case kCondNE:
698 Bne(rs, rt, imm16);
699 break;
700 case kCondEQZ:
701 CHECK_EQ(rt, ZERO);
702 Beqz(rs, imm16);
703 break;
704 case kCondNEZ:
705 CHECK_EQ(rt, ZERO);
706 Bnez(rs, imm16);
707 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800708 case kCondF:
709 CHECK_EQ(rt, ZERO);
710 Bc1f(static_cast<int>(rs), imm16);
711 break;
712 case kCondT:
713 CHECK_EQ(rt, ZERO);
714 Bc1t(static_cast<int>(rs), imm16);
715 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200716 case kCondLT:
717 case kCondGE:
718 case kCondLE:
719 case kCondGT:
720 case kCondLTU:
721 case kCondGEU:
722 case kUncond:
723 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
724 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
725 LOG(FATAL) << "Unexpected branch condition " << cond;
726 UNREACHABLE();
727 }
728}
729
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800730void MipsAssembler::EmitBcondR6(BranchCondition cond, Register rs, Register rt, uint32_t imm16_21) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200731 switch (cond) {
732 case kCondLT:
733 Bltc(rs, rt, imm16_21);
734 break;
735 case kCondGE:
736 Bgec(rs, rt, imm16_21);
737 break;
738 case kCondLE:
739 Bgec(rt, rs, imm16_21);
740 break;
741 case kCondGT:
742 Bltc(rt, rs, imm16_21);
743 break;
744 case kCondLTZ:
745 CHECK_EQ(rt, ZERO);
746 Bltzc(rs, imm16_21);
747 break;
748 case kCondGEZ:
749 CHECK_EQ(rt, ZERO);
750 Bgezc(rs, imm16_21);
751 break;
752 case kCondLEZ:
753 CHECK_EQ(rt, ZERO);
754 Blezc(rs, imm16_21);
755 break;
756 case kCondGTZ:
757 CHECK_EQ(rt, ZERO);
758 Bgtzc(rs, imm16_21);
759 break;
760 case kCondEQ:
761 Beqc(rs, rt, imm16_21);
762 break;
763 case kCondNE:
764 Bnec(rs, rt, imm16_21);
765 break;
766 case kCondEQZ:
767 CHECK_EQ(rt, ZERO);
768 Beqzc(rs, imm16_21);
769 break;
770 case kCondNEZ:
771 CHECK_EQ(rt, ZERO);
772 Bnezc(rs, imm16_21);
773 break;
774 case kCondLTU:
775 Bltuc(rs, rt, imm16_21);
776 break;
777 case kCondGEU:
778 Bgeuc(rs, rt, imm16_21);
779 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800780 case kCondF:
781 CHECK_EQ(rt, ZERO);
782 Bc1eqz(static_cast<FRegister>(rs), imm16_21);
783 break;
784 case kCondT:
785 CHECK_EQ(rt, ZERO);
786 Bc1nez(static_cast<FRegister>(rs), imm16_21);
787 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200788 case kUncond:
789 LOG(FATAL) << "Unexpected branch condition " << cond;
790 UNREACHABLE();
791 }
jeffhao7fbee072012-08-24 17:56:54 -0700792}
793
794void MipsAssembler::AddS(FRegister fd, FRegister fs, FRegister ft) {
795 EmitFR(0x11, 0x10, ft, fs, fd, 0x0);
796}
797
798void MipsAssembler::SubS(FRegister fd, FRegister fs, FRegister ft) {
799 EmitFR(0x11, 0x10, ft, fs, fd, 0x1);
800}
801
802void MipsAssembler::MulS(FRegister fd, FRegister fs, FRegister ft) {
803 EmitFR(0x11, 0x10, ft, fs, fd, 0x2);
804}
805
806void MipsAssembler::DivS(FRegister fd, FRegister fs, FRegister ft) {
807 EmitFR(0x11, 0x10, ft, fs, fd, 0x3);
808}
809
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200810void MipsAssembler::AddD(FRegister fd, FRegister fs, FRegister ft) {
811 EmitFR(0x11, 0x11, ft, fs, fd, 0x0);
jeffhao7fbee072012-08-24 17:56:54 -0700812}
813
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200814void MipsAssembler::SubD(FRegister fd, FRegister fs, FRegister ft) {
815 EmitFR(0x11, 0x11, ft, fs, fd, 0x1);
jeffhao7fbee072012-08-24 17:56:54 -0700816}
817
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200818void MipsAssembler::MulD(FRegister fd, FRegister fs, FRegister ft) {
819 EmitFR(0x11, 0x11, ft, fs, fd, 0x2);
jeffhao7fbee072012-08-24 17:56:54 -0700820}
821
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200822void MipsAssembler::DivD(FRegister fd, FRegister fs, FRegister ft) {
823 EmitFR(0x11, 0x11, ft, fs, fd, 0x3);
jeffhao7fbee072012-08-24 17:56:54 -0700824}
825
826void MipsAssembler::MovS(FRegister fd, FRegister fs) {
827 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x6);
828}
829
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +0200830void MipsAssembler::MovD(FRegister fd, FRegister fs) {
831 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x6);
832}
833
834void MipsAssembler::NegS(FRegister fd, FRegister fs) {
835 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x7);
836}
837
838void MipsAssembler::NegD(FRegister fd, FRegister fs) {
839 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x7);
840}
841
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -0800842void MipsAssembler::CunS(int cc, FRegister fs, FRegister ft) {
843 CHECK(!IsR6());
844 CHECK(IsUint<3>(cc)) << cc;
845 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x31);
846}
847
848void MipsAssembler::CeqS(int cc, FRegister fs, FRegister ft) {
849 CHECK(!IsR6());
850 CHECK(IsUint<3>(cc)) << cc;
851 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x32);
852}
853
854void MipsAssembler::CueqS(int cc, FRegister fs, FRegister ft) {
855 CHECK(!IsR6());
856 CHECK(IsUint<3>(cc)) << cc;
857 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x33);
858}
859
860void MipsAssembler::ColtS(int cc, FRegister fs, FRegister ft) {
861 CHECK(!IsR6());
862 CHECK(IsUint<3>(cc)) << cc;
863 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x34);
864}
865
866void MipsAssembler::CultS(int cc, FRegister fs, FRegister ft) {
867 CHECK(!IsR6());
868 CHECK(IsUint<3>(cc)) << cc;
869 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x35);
870}
871
872void MipsAssembler::ColeS(int cc, FRegister fs, FRegister ft) {
873 CHECK(!IsR6());
874 CHECK(IsUint<3>(cc)) << cc;
875 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x36);
876}
877
878void MipsAssembler::CuleS(int cc, FRegister fs, FRegister ft) {
879 CHECK(!IsR6());
880 CHECK(IsUint<3>(cc)) << cc;
881 EmitFR(0x11, 0x10, ft, fs, static_cast<FRegister>(cc << 2), 0x37);
882}
883
884void MipsAssembler::CunD(int cc, FRegister fs, FRegister ft) {
885 CHECK(!IsR6());
886 CHECK(IsUint<3>(cc)) << cc;
887 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x31);
888}
889
890void MipsAssembler::CeqD(int cc, FRegister fs, FRegister ft) {
891 CHECK(!IsR6());
892 CHECK(IsUint<3>(cc)) << cc;
893 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x32);
894}
895
896void MipsAssembler::CueqD(int cc, FRegister fs, FRegister ft) {
897 CHECK(!IsR6());
898 CHECK(IsUint<3>(cc)) << cc;
899 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x33);
900}
901
902void MipsAssembler::ColtD(int cc, FRegister fs, FRegister ft) {
903 CHECK(!IsR6());
904 CHECK(IsUint<3>(cc)) << cc;
905 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x34);
906}
907
908void MipsAssembler::CultD(int cc, FRegister fs, FRegister ft) {
909 CHECK(!IsR6());
910 CHECK(IsUint<3>(cc)) << cc;
911 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x35);
912}
913
914void MipsAssembler::ColeD(int cc, FRegister fs, FRegister ft) {
915 CHECK(!IsR6());
916 CHECK(IsUint<3>(cc)) << cc;
917 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x36);
918}
919
920void MipsAssembler::CuleD(int cc, FRegister fs, FRegister ft) {
921 CHECK(!IsR6());
922 CHECK(IsUint<3>(cc)) << cc;
923 EmitFR(0x11, 0x11, ft, fs, static_cast<FRegister>(cc << 2), 0x37);
924}
925
926void MipsAssembler::CmpUnS(FRegister fd, FRegister fs, FRegister ft) {
927 CHECK(IsR6());
928 EmitFR(0x11, 0x14, ft, fs, fd, 0x01);
929}
930
931void MipsAssembler::CmpEqS(FRegister fd, FRegister fs, FRegister ft) {
932 CHECK(IsR6());
933 EmitFR(0x11, 0x14, ft, fs, fd, 0x02);
934}
935
936void MipsAssembler::CmpUeqS(FRegister fd, FRegister fs, FRegister ft) {
937 CHECK(IsR6());
938 EmitFR(0x11, 0x14, ft, fs, fd, 0x03);
939}
940
941void MipsAssembler::CmpLtS(FRegister fd, FRegister fs, FRegister ft) {
942 CHECK(IsR6());
943 EmitFR(0x11, 0x14, ft, fs, fd, 0x04);
944}
945
946void MipsAssembler::CmpUltS(FRegister fd, FRegister fs, FRegister ft) {
947 CHECK(IsR6());
948 EmitFR(0x11, 0x14, ft, fs, fd, 0x05);
949}
950
951void MipsAssembler::CmpLeS(FRegister fd, FRegister fs, FRegister ft) {
952 CHECK(IsR6());
953 EmitFR(0x11, 0x14, ft, fs, fd, 0x06);
954}
955
956void MipsAssembler::CmpUleS(FRegister fd, FRegister fs, FRegister ft) {
957 CHECK(IsR6());
958 EmitFR(0x11, 0x14, ft, fs, fd, 0x07);
959}
960
961void MipsAssembler::CmpOrS(FRegister fd, FRegister fs, FRegister ft) {
962 CHECK(IsR6());
963 EmitFR(0x11, 0x14, ft, fs, fd, 0x11);
964}
965
966void MipsAssembler::CmpUneS(FRegister fd, FRegister fs, FRegister ft) {
967 CHECK(IsR6());
968 EmitFR(0x11, 0x14, ft, fs, fd, 0x12);
969}
970
971void MipsAssembler::CmpNeS(FRegister fd, FRegister fs, FRegister ft) {
972 CHECK(IsR6());
973 EmitFR(0x11, 0x14, ft, fs, fd, 0x13);
974}
975
976void MipsAssembler::CmpUnD(FRegister fd, FRegister fs, FRegister ft) {
977 CHECK(IsR6());
978 EmitFR(0x11, 0x15, ft, fs, fd, 0x01);
979}
980
981void MipsAssembler::CmpEqD(FRegister fd, FRegister fs, FRegister ft) {
982 CHECK(IsR6());
983 EmitFR(0x11, 0x15, ft, fs, fd, 0x02);
984}
985
986void MipsAssembler::CmpUeqD(FRegister fd, FRegister fs, FRegister ft) {
987 CHECK(IsR6());
988 EmitFR(0x11, 0x15, ft, fs, fd, 0x03);
989}
990
991void MipsAssembler::CmpLtD(FRegister fd, FRegister fs, FRegister ft) {
992 CHECK(IsR6());
993 EmitFR(0x11, 0x15, ft, fs, fd, 0x04);
994}
995
996void MipsAssembler::CmpUltD(FRegister fd, FRegister fs, FRegister ft) {
997 CHECK(IsR6());
998 EmitFR(0x11, 0x15, ft, fs, fd, 0x05);
999}
1000
1001void MipsAssembler::CmpLeD(FRegister fd, FRegister fs, FRegister ft) {
1002 CHECK(IsR6());
1003 EmitFR(0x11, 0x15, ft, fs, fd, 0x06);
1004}
1005
1006void MipsAssembler::CmpUleD(FRegister fd, FRegister fs, FRegister ft) {
1007 CHECK(IsR6());
1008 EmitFR(0x11, 0x15, ft, fs, fd, 0x07);
1009}
1010
1011void MipsAssembler::CmpOrD(FRegister fd, FRegister fs, FRegister ft) {
1012 CHECK(IsR6());
1013 EmitFR(0x11, 0x15, ft, fs, fd, 0x11);
1014}
1015
1016void MipsAssembler::CmpUneD(FRegister fd, FRegister fs, FRegister ft) {
1017 CHECK(IsR6());
1018 EmitFR(0x11, 0x15, ft, fs, fd, 0x12);
1019}
1020
1021void MipsAssembler::CmpNeD(FRegister fd, FRegister fs, FRegister ft) {
1022 CHECK(IsR6());
1023 EmitFR(0x11, 0x15, ft, fs, fd, 0x13);
1024}
1025
1026void MipsAssembler::Movf(Register rd, Register rs, int cc) {
1027 CHECK(!IsR6());
1028 CHECK(IsUint<3>(cc)) << cc;
1029 EmitR(0, rs, static_cast<Register>(cc << 2), rd, 0, 0x01);
1030}
1031
1032void MipsAssembler::Movt(Register rd, Register rs, int cc) {
1033 CHECK(!IsR6());
1034 CHECK(IsUint<3>(cc)) << cc;
1035 EmitR(0, rs, static_cast<Register>((cc << 2) | 1), rd, 0, 0x01);
1036}
1037
Alexey Frunzebaf60b72015-12-22 15:15:03 -08001038void MipsAssembler::TruncLS(FRegister fd, FRegister fs) {
1039 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x09);
1040}
1041
1042void MipsAssembler::TruncLD(FRegister fd, FRegister fs) {
1043 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x09);
1044}
1045
1046void MipsAssembler::TruncWS(FRegister fd, FRegister fs) {
1047 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x0D);
1048}
1049
1050void MipsAssembler::TruncWD(FRegister fd, FRegister fs) {
1051 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x0D);
1052}
1053
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001054void MipsAssembler::Cvtsw(FRegister fd, FRegister fs) {
1055 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x20);
1056}
1057
1058void MipsAssembler::Cvtdw(FRegister fd, FRegister fs) {
1059 EmitFR(0x11, 0x14, static_cast<FRegister>(0), fs, fd, 0x21);
1060}
1061
1062void MipsAssembler::Cvtsd(FRegister fd, FRegister fs) {
1063 EmitFR(0x11, 0x11, static_cast<FRegister>(0), fs, fd, 0x20);
1064}
1065
1066void MipsAssembler::Cvtds(FRegister fd, FRegister fs) {
1067 EmitFR(0x11, 0x10, static_cast<FRegister>(0), fs, fd, 0x21);
jeffhao7fbee072012-08-24 17:56:54 -07001068}
1069
Alexey Frunzebaf60b72015-12-22 15:15:03 -08001070void MipsAssembler::Cvtsl(FRegister fd, FRegister fs) {
1071 EmitFR(0x11, 0x15, static_cast<FRegister>(0), fs, fd, 0x20);
1072}
1073
1074void MipsAssembler::Cvtdl(FRegister fd, FRegister fs) {
1075 EmitFR(0x11, 0x15, static_cast<FRegister>(0), fs, fd, 0x21);
1076}
1077
jeffhao7fbee072012-08-24 17:56:54 -07001078void MipsAssembler::Mfc1(Register rt, FRegister fs) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001079 EmitFR(0x11, 0x00, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -07001080}
1081
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001082void MipsAssembler::Mtc1(Register rt, FRegister fs) {
1083 EmitFR(0x11, 0x04, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
1084}
1085
1086void MipsAssembler::Mfhc1(Register rt, FRegister fs) {
1087 EmitFR(0x11, 0x03, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
1088}
1089
1090void MipsAssembler::Mthc1(Register rt, FRegister fs) {
1091 EmitFR(0x11, 0x07, static_cast<FRegister>(rt), fs, static_cast<FRegister>(0), 0x0);
jeffhao7fbee072012-08-24 17:56:54 -07001092}
1093
Alexey Frunzebb9863a2016-01-11 15:51:16 -08001094void MipsAssembler::MoveFromFpuHigh(Register rt, FRegister fs) {
1095 if (Is32BitFPU()) {
1096 CHECK_EQ(fs % 2, 0) << fs;
1097 Mfc1(rt, static_cast<FRegister>(fs + 1));
1098 } else {
1099 Mfhc1(rt, fs);
1100 }
1101}
1102
1103void MipsAssembler::MoveToFpuHigh(Register rt, FRegister fs) {
1104 if (Is32BitFPU()) {
1105 CHECK_EQ(fs % 2, 0) << fs;
1106 Mtc1(rt, static_cast<FRegister>(fs + 1));
1107 } else {
1108 Mthc1(rt, fs);
1109 }
1110}
1111
jeffhao7fbee072012-08-24 17:56:54 -07001112void MipsAssembler::Lwc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001113 EmitI(0x31, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001114}
1115
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001116void MipsAssembler::Ldc1(FRegister ft, Register rs, uint16_t imm16) {
1117 EmitI(0x35, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001118}
1119
1120void MipsAssembler::Swc1(FRegister ft, Register rs, uint16_t imm16) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001121 EmitI(0x39, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001122}
1123
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001124void MipsAssembler::Sdc1(FRegister ft, Register rs, uint16_t imm16) {
1125 EmitI(0x3d, rs, static_cast<Register>(ft), imm16);
jeffhao7fbee072012-08-24 17:56:54 -07001126}
1127
1128void MipsAssembler::Break() {
1129 EmitR(0, static_cast<Register>(0), static_cast<Register>(0),
1130 static_cast<Register>(0), 0, 0xD);
1131}
1132
jeffhao07030602012-09-26 14:33:14 -07001133void MipsAssembler::Nop() {
1134 EmitR(0x0, static_cast<Register>(0), static_cast<Register>(0), static_cast<Register>(0), 0, 0x0);
1135}
1136
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001137void MipsAssembler::Move(Register rd, Register rs) {
1138 Or(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001139}
1140
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001141void MipsAssembler::Clear(Register rd) {
1142 Move(rd, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001143}
1144
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001145void MipsAssembler::Not(Register rd, Register rs) {
1146 Nor(rd, rs, ZERO);
jeffhao7fbee072012-08-24 17:56:54 -07001147}
1148
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001149void MipsAssembler::Push(Register rs) {
1150 IncreaseFrameSize(kMipsWordSize);
1151 Sw(rs, SP, 0);
jeffhao7fbee072012-08-24 17:56:54 -07001152}
1153
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001154void MipsAssembler::Pop(Register rd) {
1155 Lw(rd, SP, 0);
1156 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001157}
1158
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001159void MipsAssembler::PopAndReturn(Register rd, Register rt) {
1160 Lw(rd, SP, 0);
1161 Jr(rt);
1162 DecreaseFrameSize(kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07001163}
1164
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001165void MipsAssembler::LoadConst32(Register rd, int32_t value) {
1166 if (IsUint<16>(value)) {
1167 // Use OR with (unsigned) immediate to encode 16b unsigned int.
1168 Ori(rd, ZERO, value);
1169 } else if (IsInt<16>(value)) {
1170 // Use ADD with (signed) immediate to encode 16b signed int.
1171 Addiu(rd, ZERO, value);
jeffhao7fbee072012-08-24 17:56:54 -07001172 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001173 Lui(rd, High16Bits(value));
1174 if (value & 0xFFFF)
1175 Ori(rd, rd, Low16Bits(value));
1176 }
1177}
1178
1179void MipsAssembler::LoadConst64(Register reg_hi, Register reg_lo, int64_t value) {
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001180 uint32_t low = Low32Bits(value);
1181 uint32_t high = High32Bits(value);
1182 LoadConst32(reg_lo, low);
1183 if (high != low) {
1184 LoadConst32(reg_hi, high);
1185 } else {
1186 Move(reg_hi, reg_lo);
1187 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001188}
1189
1190void MipsAssembler::StoreConst32ToOffset(int32_t value,
1191 Register base,
1192 int32_t offset,
1193 Register temp) {
1194 if (!IsInt<16>(offset)) {
1195 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
1196 LoadConst32(AT, offset);
1197 Addu(AT, AT, base);
1198 base = AT;
1199 offset = 0;
1200 }
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001201 if (value == 0) {
1202 temp = ZERO;
1203 } else {
1204 LoadConst32(temp, value);
1205 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001206 Sw(temp, base, offset);
1207}
1208
1209void MipsAssembler::StoreConst64ToOffset(int64_t value,
1210 Register base,
1211 int32_t offset,
1212 Register temp) {
1213 // IsInt<16> must be passed a signed value.
1214 if (!IsInt<16>(offset) || !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize))) {
1215 CHECK_NE(temp, AT); // Must not use AT as temp, as not to overwrite the loaded value.
1216 LoadConst32(AT, offset);
1217 Addu(AT, AT, base);
1218 base = AT;
1219 offset = 0;
1220 }
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001221 uint32_t low = Low32Bits(value);
1222 uint32_t high = High32Bits(value);
1223 if (low == 0) {
1224 Sw(ZERO, base, offset);
1225 } else {
1226 LoadConst32(temp, low);
1227 Sw(temp, base, offset);
1228 }
1229 if (high == 0) {
1230 Sw(ZERO, base, offset + kMipsWordSize);
1231 } else {
1232 if (high != low) {
1233 LoadConst32(temp, high);
1234 }
1235 Sw(temp, base, offset + kMipsWordSize);
1236 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001237}
1238
1239void MipsAssembler::LoadSConst32(FRegister r, int32_t value, Register temp) {
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001240 if (value == 0) {
1241 temp = ZERO;
1242 } else {
1243 LoadConst32(temp, value);
1244 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001245 Mtc1(temp, r);
1246}
1247
1248void MipsAssembler::LoadDConst64(FRegister rd, int64_t value, Register temp) {
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001249 uint32_t low = Low32Bits(value);
1250 uint32_t high = High32Bits(value);
1251 if (low == 0) {
1252 Mtc1(ZERO, rd);
1253 } else {
1254 LoadConst32(temp, low);
1255 Mtc1(temp, rd);
1256 }
1257 if (high == 0) {
Alexey Frunzebb9863a2016-01-11 15:51:16 -08001258 MoveToFpuHigh(ZERO, rd);
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001259 } else {
1260 LoadConst32(temp, high);
Alexey Frunzebb9863a2016-01-11 15:51:16 -08001261 MoveToFpuHigh(temp, rd);
Alexey Frunze5c7aed32015-11-25 19:41:54 -08001262 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001263}
1264
1265void MipsAssembler::Addiu32(Register rt, Register rs, int32_t value, Register temp) {
1266 if (IsInt<16>(value)) {
1267 Addiu(rt, rs, value);
1268 } else {
1269 LoadConst32(temp, value);
1270 Addu(rt, rs, temp);
1271 }
1272}
1273
1274void MipsAssembler::Branch::InitShortOrLong(MipsAssembler::Branch::OffsetBits offset_size,
1275 MipsAssembler::Branch::Type short_type,
1276 MipsAssembler::Branch::Type long_type) {
1277 type_ = (offset_size <= branch_info_[short_type].offset_size) ? short_type : long_type;
1278}
1279
1280void MipsAssembler::Branch::InitializeType(bool is_call, bool is_r6) {
1281 OffsetBits offset_size = GetOffsetSizeNeeded(location_, target_);
1282 if (is_r6) {
1283 // R6
1284 if (is_call) {
1285 InitShortOrLong(offset_size, kR6Call, kR6LongCall);
1286 } else if (condition_ == kUncond) {
1287 InitShortOrLong(offset_size, kR6UncondBranch, kR6LongUncondBranch);
1288 } else {
1289 if (condition_ == kCondEQZ || condition_ == kCondNEZ) {
1290 // Special case for beqzc/bnezc with longer offset than in other b<cond>c instructions.
1291 type_ = (offset_size <= kOffset23) ? kR6CondBranch : kR6LongCondBranch;
1292 } else {
1293 InitShortOrLong(offset_size, kR6CondBranch, kR6LongCondBranch);
1294 }
1295 }
1296 } else {
1297 // R2
1298 if (is_call) {
1299 InitShortOrLong(offset_size, kCall, kLongCall);
1300 } else if (condition_ == kUncond) {
1301 InitShortOrLong(offset_size, kUncondBranch, kLongUncondBranch);
1302 } else {
1303 InitShortOrLong(offset_size, kCondBranch, kLongCondBranch);
1304 }
1305 }
1306 old_type_ = type_;
1307}
1308
1309bool MipsAssembler::Branch::IsNop(BranchCondition condition, Register lhs, Register rhs) {
1310 switch (condition) {
1311 case kCondLT:
1312 case kCondGT:
1313 case kCondNE:
1314 case kCondLTU:
1315 return lhs == rhs;
1316 default:
1317 return false;
1318 }
1319}
1320
1321bool MipsAssembler::Branch::IsUncond(BranchCondition condition, Register lhs, Register rhs) {
1322 switch (condition) {
1323 case kUncond:
1324 return true;
1325 case kCondGE:
1326 case kCondLE:
1327 case kCondEQ:
1328 case kCondGEU:
1329 return lhs == rhs;
1330 default:
1331 return false;
1332 }
1333}
1334
1335MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target)
1336 : old_location_(location),
1337 location_(location),
1338 target_(target),
1339 lhs_reg_(0),
1340 rhs_reg_(0),
1341 condition_(kUncond) {
1342 InitializeType(false, is_r6);
1343}
1344
1345MipsAssembler::Branch::Branch(bool is_r6,
1346 uint32_t location,
1347 uint32_t target,
1348 MipsAssembler::BranchCondition condition,
1349 Register lhs_reg,
1350 Register rhs_reg)
1351 : old_location_(location),
1352 location_(location),
1353 target_(target),
1354 lhs_reg_(lhs_reg),
1355 rhs_reg_(rhs_reg),
1356 condition_(condition) {
1357 CHECK_NE(condition, kUncond);
1358 switch (condition) {
1359 case kCondLT:
1360 case kCondGE:
1361 case kCondLE:
1362 case kCondGT:
1363 case kCondLTU:
1364 case kCondGEU:
1365 // We don't support synthetic R2 branches (preceded with slt[u]) at this level
1366 // (R2 doesn't have branches to compare 2 registers using <, <=, >=, >).
1367 // We leave this up to the caller.
1368 CHECK(is_r6);
1369 FALLTHROUGH_INTENDED;
1370 case kCondEQ:
1371 case kCondNE:
1372 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1373 // To compare with 0, use dedicated kCond*Z conditions.
1374 CHECK_NE(lhs_reg, ZERO);
1375 CHECK_NE(rhs_reg, ZERO);
1376 break;
1377 case kCondLTZ:
1378 case kCondGEZ:
1379 case kCondLEZ:
1380 case kCondGTZ:
1381 case kCondEQZ:
1382 case kCondNEZ:
1383 // Require registers other than 0 not only for R6, but also for R2 to catch errors.
1384 CHECK_NE(lhs_reg, ZERO);
1385 CHECK_EQ(rhs_reg, ZERO);
1386 break;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001387 case kCondF:
1388 case kCondT:
1389 CHECK_EQ(rhs_reg, ZERO);
1390 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001391 case kUncond:
1392 UNREACHABLE();
1393 }
1394 CHECK(!IsNop(condition, lhs_reg, rhs_reg));
1395 if (IsUncond(condition, lhs_reg, rhs_reg)) {
1396 // Branch condition is always true, make the branch unconditional.
1397 condition_ = kUncond;
1398 }
1399 InitializeType(false, is_r6);
1400}
1401
1402MipsAssembler::Branch::Branch(bool is_r6, uint32_t location, uint32_t target, Register indirect_reg)
1403 : old_location_(location),
1404 location_(location),
1405 target_(target),
1406 lhs_reg_(indirect_reg),
1407 rhs_reg_(0),
1408 condition_(kUncond) {
1409 CHECK_NE(indirect_reg, ZERO);
1410 CHECK_NE(indirect_reg, AT);
1411 InitializeType(true, is_r6);
1412}
1413
1414MipsAssembler::BranchCondition MipsAssembler::Branch::OppositeCondition(
1415 MipsAssembler::BranchCondition cond) {
1416 switch (cond) {
1417 case kCondLT:
1418 return kCondGE;
1419 case kCondGE:
1420 return kCondLT;
1421 case kCondLE:
1422 return kCondGT;
1423 case kCondGT:
1424 return kCondLE;
1425 case kCondLTZ:
1426 return kCondGEZ;
1427 case kCondGEZ:
1428 return kCondLTZ;
1429 case kCondLEZ:
1430 return kCondGTZ;
1431 case kCondGTZ:
1432 return kCondLEZ;
1433 case kCondEQ:
1434 return kCondNE;
1435 case kCondNE:
1436 return kCondEQ;
1437 case kCondEQZ:
1438 return kCondNEZ;
1439 case kCondNEZ:
1440 return kCondEQZ;
1441 case kCondLTU:
1442 return kCondGEU;
1443 case kCondGEU:
1444 return kCondLTU;
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001445 case kCondF:
1446 return kCondT;
1447 case kCondT:
1448 return kCondF;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001449 case kUncond:
1450 LOG(FATAL) << "Unexpected branch condition " << cond;
1451 }
1452 UNREACHABLE();
1453}
1454
1455MipsAssembler::Branch::Type MipsAssembler::Branch::GetType() const {
1456 return type_;
1457}
1458
1459MipsAssembler::BranchCondition MipsAssembler::Branch::GetCondition() const {
1460 return condition_;
1461}
1462
1463Register MipsAssembler::Branch::GetLeftRegister() const {
1464 return static_cast<Register>(lhs_reg_);
1465}
1466
1467Register MipsAssembler::Branch::GetRightRegister() const {
1468 return static_cast<Register>(rhs_reg_);
1469}
1470
1471uint32_t MipsAssembler::Branch::GetTarget() const {
1472 return target_;
1473}
1474
1475uint32_t MipsAssembler::Branch::GetLocation() const {
1476 return location_;
1477}
1478
1479uint32_t MipsAssembler::Branch::GetOldLocation() const {
1480 return old_location_;
1481}
1482
1483uint32_t MipsAssembler::Branch::GetLength() const {
1484 return branch_info_[type_].length;
1485}
1486
1487uint32_t MipsAssembler::Branch::GetOldLength() const {
1488 return branch_info_[old_type_].length;
1489}
1490
1491uint32_t MipsAssembler::Branch::GetSize() const {
1492 return GetLength() * sizeof(uint32_t);
1493}
1494
1495uint32_t MipsAssembler::Branch::GetOldSize() const {
1496 return GetOldLength() * sizeof(uint32_t);
1497}
1498
1499uint32_t MipsAssembler::Branch::GetEndLocation() const {
1500 return GetLocation() + GetSize();
1501}
1502
1503uint32_t MipsAssembler::Branch::GetOldEndLocation() const {
1504 return GetOldLocation() + GetOldSize();
1505}
1506
1507bool MipsAssembler::Branch::IsLong() const {
1508 switch (type_) {
1509 // R2 short branches.
1510 case kUncondBranch:
1511 case kCondBranch:
1512 case kCall:
1513 // R6 short branches.
1514 case kR6UncondBranch:
1515 case kR6CondBranch:
1516 case kR6Call:
1517 return false;
1518 // R2 long branches.
1519 case kLongUncondBranch:
1520 case kLongCondBranch:
1521 case kLongCall:
1522 // R6 long branches.
1523 case kR6LongUncondBranch:
1524 case kR6LongCondBranch:
1525 case kR6LongCall:
1526 return true;
1527 }
1528 UNREACHABLE();
1529}
1530
1531bool MipsAssembler::Branch::IsResolved() const {
1532 return target_ != kUnresolved;
1533}
1534
1535MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSize() const {
1536 OffsetBits offset_size =
1537 (type_ == kR6CondBranch && (condition_ == kCondEQZ || condition_ == kCondNEZ))
1538 ? kOffset23
1539 : branch_info_[type_].offset_size;
1540 return offset_size;
1541}
1542
1543MipsAssembler::Branch::OffsetBits MipsAssembler::Branch::GetOffsetSizeNeeded(uint32_t location,
1544 uint32_t target) {
1545 // For unresolved targets assume the shortest encoding
1546 // (later it will be made longer if needed).
1547 if (target == kUnresolved)
1548 return kOffset16;
1549 int64_t distance = static_cast<int64_t>(target) - location;
1550 // To simplify calculations in composite branches consisting of multiple instructions
1551 // bump up the distance by a value larger than the max byte size of a composite branch.
1552 distance += (distance >= 0) ? kMaxBranchSize : -kMaxBranchSize;
1553 if (IsInt<kOffset16>(distance))
1554 return kOffset16;
1555 else if (IsInt<kOffset18>(distance))
1556 return kOffset18;
1557 else if (IsInt<kOffset21>(distance))
1558 return kOffset21;
1559 else if (IsInt<kOffset23>(distance))
1560 return kOffset23;
1561 else if (IsInt<kOffset28>(distance))
1562 return kOffset28;
1563 return kOffset32;
1564}
1565
1566void MipsAssembler::Branch::Resolve(uint32_t target) {
1567 target_ = target;
1568}
1569
1570void MipsAssembler::Branch::Relocate(uint32_t expand_location, uint32_t delta) {
1571 if (location_ > expand_location) {
1572 location_ += delta;
1573 }
1574 if (!IsResolved()) {
1575 return; // Don't know the target yet.
1576 }
1577 if (target_ > expand_location) {
1578 target_ += delta;
1579 }
1580}
1581
1582void MipsAssembler::Branch::PromoteToLong() {
1583 switch (type_) {
1584 // R2 short branches.
1585 case kUncondBranch:
1586 type_ = kLongUncondBranch;
1587 break;
1588 case kCondBranch:
1589 type_ = kLongCondBranch;
1590 break;
1591 case kCall:
1592 type_ = kLongCall;
1593 break;
1594 // R6 short branches.
1595 case kR6UncondBranch:
1596 type_ = kR6LongUncondBranch;
1597 break;
1598 case kR6CondBranch:
1599 type_ = kR6LongCondBranch;
1600 break;
1601 case kR6Call:
1602 type_ = kR6LongCall;
1603 break;
1604 default:
1605 // Note: 'type_' is already long.
1606 break;
1607 }
1608 CHECK(IsLong());
1609}
1610
1611uint32_t MipsAssembler::Branch::PromoteIfNeeded(uint32_t max_short_distance) {
1612 // If the branch is still unresolved or already long, nothing to do.
1613 if (IsLong() || !IsResolved()) {
1614 return 0;
1615 }
1616 // Promote the short branch to long if the offset size is too small
1617 // to hold the distance between location_ and target_.
1618 if (GetOffsetSizeNeeded(location_, target_) > GetOffsetSize()) {
1619 PromoteToLong();
1620 uint32_t old_size = GetOldSize();
1621 uint32_t new_size = GetSize();
1622 CHECK_GT(new_size, old_size);
1623 return new_size - old_size;
1624 }
1625 // The following logic is for debugging/testing purposes.
1626 // Promote some short branches to long when it's not really required.
1627 if (UNLIKELY(max_short_distance != std::numeric_limits<uint32_t>::max())) {
1628 int64_t distance = static_cast<int64_t>(target_) - location_;
1629 distance = (distance >= 0) ? distance : -distance;
1630 if (distance >= max_short_distance) {
1631 PromoteToLong();
1632 uint32_t old_size = GetOldSize();
1633 uint32_t new_size = GetSize();
1634 CHECK_GT(new_size, old_size);
1635 return new_size - old_size;
1636 }
1637 }
1638 return 0;
1639}
1640
1641uint32_t MipsAssembler::Branch::GetOffsetLocation() const {
1642 return location_ + branch_info_[type_].instr_offset * sizeof(uint32_t);
1643}
1644
1645uint32_t MipsAssembler::Branch::GetOffset() const {
1646 CHECK(IsResolved());
1647 uint32_t ofs_mask = 0xFFFFFFFF >> (32 - GetOffsetSize());
1648 // Calculate the byte distance between instructions and also account for
1649 // different PC-relative origins.
1650 uint32_t offset = target_ - GetOffsetLocation() - branch_info_[type_].pc_org * sizeof(uint32_t);
1651 // Prepare the offset for encoding into the instruction(s).
1652 offset = (offset & ofs_mask) >> branch_info_[type_].offset_shift;
1653 return offset;
1654}
1655
1656MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) {
1657 CHECK_LT(branch_id, branches_.size());
1658 return &branches_[branch_id];
1659}
1660
1661const MipsAssembler::Branch* MipsAssembler::GetBranch(uint32_t branch_id) const {
1662 CHECK_LT(branch_id, branches_.size());
1663 return &branches_[branch_id];
1664}
1665
1666void MipsAssembler::Bind(MipsLabel* label) {
1667 CHECK(!label->IsBound());
1668 uint32_t bound_pc = buffer_.Size();
1669
1670 // Walk the list of branches referring to and preceding this label.
1671 // Store the previously unknown target addresses in them.
1672 while (label->IsLinked()) {
1673 uint32_t branch_id = label->Position();
1674 Branch* branch = GetBranch(branch_id);
1675 branch->Resolve(bound_pc);
1676
1677 uint32_t branch_location = branch->GetLocation();
1678 // Extract the location of the previous branch in the list (walking the list backwards;
1679 // the previous branch ID was stored in the space reserved for this branch).
1680 uint32_t prev = buffer_.Load<uint32_t>(branch_location);
1681
1682 // On to the previous branch in the list...
1683 label->position_ = prev;
1684 }
1685
1686 // Now make the label object contain its own location (relative to the end of the preceding
1687 // branch, if any; it will be used by the branches referring to and following this label).
1688 label->prev_branch_id_plus_one_ = branches_.size();
1689 if (label->prev_branch_id_plus_one_) {
1690 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1691 const Branch* branch = GetBranch(branch_id);
1692 bound_pc -= branch->GetEndLocation();
1693 }
1694 label->BindTo(bound_pc);
1695}
1696
1697uint32_t MipsAssembler::GetLabelLocation(MipsLabel* label) const {
1698 CHECK(label->IsBound());
1699 uint32_t target = label->Position();
1700 if (label->prev_branch_id_plus_one_) {
1701 // Get label location based on the branch preceding it.
1702 uint32_t branch_id = label->prev_branch_id_plus_one_ - 1;
1703 const Branch* branch = GetBranch(branch_id);
1704 target += branch->GetEndLocation();
1705 }
1706 return target;
1707}
1708
1709uint32_t MipsAssembler::GetAdjustedPosition(uint32_t old_position) {
1710 // We can reconstruct the adjustment by going through all the branches from the beginning
1711 // up to the old_position. Since we expect AdjustedPosition() to be called in a loop
1712 // with increasing old_position, we can use the data from last AdjustedPosition() to
1713 // continue where we left off and the whole loop should be O(m+n) where m is the number
1714 // of positions to adjust and n is the number of branches.
1715 if (old_position < last_old_position_) {
1716 last_position_adjustment_ = 0;
1717 last_old_position_ = 0;
1718 last_branch_id_ = 0;
1719 }
1720 while (last_branch_id_ != branches_.size()) {
1721 const Branch* branch = GetBranch(last_branch_id_);
1722 if (branch->GetLocation() >= old_position + last_position_adjustment_) {
1723 break;
1724 }
1725 last_position_adjustment_ += branch->GetSize() - branch->GetOldSize();
1726 ++last_branch_id_;
1727 }
1728 last_old_position_ = old_position;
1729 return old_position + last_position_adjustment_;
1730}
1731
1732void MipsAssembler::FinalizeLabeledBranch(MipsLabel* label) {
1733 uint32_t length = branches_.back().GetLength();
1734 if (!label->IsBound()) {
1735 // Branch forward (to a following label), distance is unknown.
1736 // The first branch forward will contain 0, serving as the terminator of
1737 // the list of forward-reaching branches.
1738 Emit(label->position_);
1739 length--;
1740 // Now make the label object point to this branch
1741 // (this forms a linked list of branches preceding this label).
1742 uint32_t branch_id = branches_.size() - 1;
1743 label->LinkTo(branch_id);
1744 }
1745 // Reserve space for the branch.
1746 while (length--) {
1747 Nop();
1748 }
1749}
1750
1751void MipsAssembler::Buncond(MipsLabel* label) {
1752 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1753 branches_.emplace_back(IsR6(), buffer_.Size(), target);
1754 FinalizeLabeledBranch(label);
1755}
1756
1757void MipsAssembler::Bcond(MipsLabel* label, BranchCondition condition, Register lhs, Register rhs) {
1758 // If lhs = rhs, this can be a NOP.
1759 if (Branch::IsNop(condition, lhs, rhs)) {
1760 return;
1761 }
1762 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1763 branches_.emplace_back(IsR6(), buffer_.Size(), target, condition, lhs, rhs);
1764 FinalizeLabeledBranch(label);
1765}
1766
1767void MipsAssembler::Call(MipsLabel* label, Register indirect_reg) {
1768 uint32_t target = label->IsBound() ? GetLabelLocation(label) : Branch::kUnresolved;
1769 branches_.emplace_back(IsR6(), buffer_.Size(), target, indirect_reg);
1770 FinalizeLabeledBranch(label);
1771}
1772
1773void MipsAssembler::PromoteBranches() {
1774 // Promote short branches to long as necessary.
1775 bool changed;
1776 do {
1777 changed = false;
1778 for (auto& branch : branches_) {
1779 CHECK(branch.IsResolved());
1780 uint32_t delta = branch.PromoteIfNeeded();
1781 // If this branch has been promoted and needs to expand in size,
1782 // relocate all branches by the expansion size.
1783 if (delta) {
1784 changed = true;
1785 uint32_t expand_location = branch.GetLocation();
1786 for (auto& branch2 : branches_) {
1787 branch2.Relocate(expand_location, delta);
1788 }
1789 }
1790 }
1791 } while (changed);
1792
1793 // Account for branch expansion by resizing the code buffer
1794 // and moving the code in it to its final location.
1795 size_t branch_count = branches_.size();
1796 if (branch_count > 0) {
1797 // Resize.
1798 Branch& last_branch = branches_[branch_count - 1];
1799 uint32_t size_delta = last_branch.GetEndLocation() - last_branch.GetOldEndLocation();
1800 uint32_t old_size = buffer_.Size();
1801 buffer_.Resize(old_size + size_delta);
1802 // Move the code residing between branch placeholders.
1803 uint32_t end = old_size;
1804 for (size_t i = branch_count; i > 0; ) {
1805 Branch& branch = branches_[--i];
1806 uint32_t size = end - branch.GetOldEndLocation();
1807 buffer_.Move(branch.GetEndLocation(), branch.GetOldEndLocation(), size);
1808 end = branch.GetOldLocation();
1809 }
1810 }
1811}
1812
1813// Note: make sure branch_info_[] and EmitBranch() are kept synchronized.
1814const MipsAssembler::Branch::BranchInfo MipsAssembler::Branch::branch_info_[] = {
1815 // R2 short branches.
1816 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kUncondBranch
1817 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kCondBranch
1818 { 5, 2, 0, MipsAssembler::Branch::kOffset16, 0 }, // kCall
1819 // R2 long branches.
1820 { 9, 3, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongUncondBranch
1821 { 10, 4, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCondBranch
1822 { 6, 1, 1, MipsAssembler::Branch::kOffset32, 0 }, // kLongCall
1823 // R6 short branches.
1824 { 1, 0, 1, MipsAssembler::Branch::kOffset28, 2 }, // kR6UncondBranch
1825 { 2, 0, 1, MipsAssembler::Branch::kOffset18, 2 }, // kR6CondBranch
1826 // Exception: kOffset23 for beqzc/bnezc.
1827 { 2, 0, 0, MipsAssembler::Branch::kOffset21, 2 }, // kR6Call
1828 // R6 long branches.
1829 { 2, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongUncondBranch
1830 { 3, 1, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCondBranch
1831 { 3, 0, 0, MipsAssembler::Branch::kOffset32, 0 }, // kR6LongCall
1832};
1833
1834// Note: make sure branch_info_[] and mitBranch() are kept synchronized.
1835void MipsAssembler::EmitBranch(MipsAssembler::Branch* branch) {
1836 CHECK_EQ(overwriting_, true);
1837 overwrite_location_ = branch->GetLocation();
1838 uint32_t offset = branch->GetOffset();
1839 BranchCondition condition = branch->GetCondition();
1840 Register lhs = branch->GetLeftRegister();
1841 Register rhs = branch->GetRightRegister();
1842 switch (branch->GetType()) {
1843 // R2 short branches.
1844 case Branch::kUncondBranch:
1845 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1846 B(offset);
1847 Nop(); // TODO: improve by filling the delay slot.
1848 break;
1849 case Branch::kCondBranch:
1850 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001851 EmitBcondR2(condition, lhs, rhs, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001852 Nop(); // TODO: improve by filling the delay slot.
1853 break;
1854 case Branch::kCall:
1855 Nal();
1856 Nop(); // TODO: is this NOP really needed here?
1857 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1858 Addiu(lhs, RA, offset);
1859 Jalr(lhs);
1860 Nop();
1861 break;
1862
1863 // R2 long branches.
1864 case Branch::kLongUncondBranch:
1865 // To get the value of the PC register we need to use the NAL instruction.
1866 // NAL clobbers the RA register. However, RA must be preserved if the
1867 // method is compiled without the entry/exit sequences that would take care
1868 // of preserving RA (typically, leaf methods don't preserve RA explicitly).
1869 // So, we need to preserve RA in some temporary storage ourselves. The AT
1870 // register can't be used for this because we need it to load a constant
1871 // which will be added to the value that NAL stores in RA. And we can't
1872 // use T9 for this in the context of the JNI compiler, which uses it
1873 // as a scratch register (see InterproceduralScratchRegister()).
1874 // If we were to add a 32-bit constant to RA using two ADDIU instructions,
1875 // we'd also need to use the ROTR instruction, which requires no less than
1876 // MIPSR2.
1877 // Perhaps, we could use T8 or one of R2's multiplier/divider registers
1878 // (LO or HI) or even a floating-point register, but that doesn't seem
1879 // like a nice solution. We may want this to work on both R6 and pre-R6.
1880 // For now simply use the stack for RA. This should be OK since for the
1881 // vast majority of code a short PC-relative branch is sufficient.
1882 // TODO: can this be improved?
1883 Push(RA);
1884 Nal();
1885 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1886 Lui(AT, High16Bits(offset));
1887 Ori(AT, AT, Low16Bits(offset));
1888 Addu(AT, AT, RA);
1889 Lw(RA, SP, 0);
1890 Jr(AT);
1891 DecreaseFrameSize(kMipsWordSize);
1892 break;
1893 case Branch::kLongCondBranch:
1894 // The comment on case 'Branch::kLongUncondBranch' applies here as well.
1895 // Note: the opposite condition branch encodes 8 as the distance, which is equal to the
1896 // number of instructions skipped:
1897 // (PUSH(IncreaseFrameSize(ADDIU) + SW) + NAL + LUI + ORI + ADDU + LW + JR).
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001898 EmitBcondR2(Branch::OppositeCondition(condition), lhs, rhs, 8);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001899 Push(RA);
1900 Nal();
1901 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1902 Lui(AT, High16Bits(offset));
1903 Ori(AT, AT, Low16Bits(offset));
1904 Addu(AT, AT, RA);
1905 Lw(RA, SP, 0);
1906 Jr(AT);
1907 DecreaseFrameSize(kMipsWordSize);
1908 break;
1909 case Branch::kLongCall:
1910 Nal();
1911 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1912 Lui(AT, High16Bits(offset));
1913 Ori(AT, AT, Low16Bits(offset));
1914 Addu(lhs, AT, RA);
1915 Jalr(lhs);
1916 Nop();
1917 break;
1918
1919 // R6 short branches.
1920 case Branch::kR6UncondBranch:
1921 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1922 Bc(offset);
1923 break;
1924 case Branch::kR6CondBranch:
1925 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001926 EmitBcondR6(condition, lhs, rhs, offset);
1927 Nop(); // TODO: improve by filling the forbidden/delay slot.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001928 break;
1929 case Branch::kR6Call:
1930 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1931 Addiupc(lhs, offset);
1932 Jialc(lhs, 0);
1933 break;
1934
1935 // R6 long branches.
1936 case Branch::kR6LongUncondBranch:
1937 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1938 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1939 Auipc(AT, High16Bits(offset));
1940 Jic(AT, Low16Bits(offset));
1941 break;
1942 case Branch::kR6LongCondBranch:
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08001943 EmitBcondR6(Branch::OppositeCondition(condition), lhs, rhs, 2);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02001944 offset += (offset & 0x8000) << 1; // Account for sign extension in jic.
1945 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1946 Auipc(AT, High16Bits(offset));
1947 Jic(AT, Low16Bits(offset));
1948 break;
1949 case Branch::kR6LongCall:
1950 offset += (offset & 0x8000) << 1; // Account for sign extension in addiu.
1951 CHECK_EQ(overwrite_location_, branch->GetOffsetLocation());
1952 Auipc(lhs, High16Bits(offset));
1953 Addiu(lhs, lhs, Low16Bits(offset));
1954 Jialc(lhs, 0);
1955 break;
1956 }
1957 CHECK_EQ(overwrite_location_, branch->GetEndLocation());
1958 CHECK_LT(branch->GetSize(), static_cast<uint32_t>(Branch::kMaxBranchSize));
1959}
1960
1961void MipsAssembler::B(MipsLabel* label) {
1962 Buncond(label);
1963}
1964
1965void MipsAssembler::Jalr(MipsLabel* label, Register indirect_reg) {
1966 Call(label, indirect_reg);
1967}
1968
1969void MipsAssembler::Beq(Register rs, Register rt, MipsLabel* label) {
1970 Bcond(label, kCondEQ, rs, rt);
1971}
1972
1973void MipsAssembler::Bne(Register rs, Register rt, MipsLabel* label) {
1974 Bcond(label, kCondNE, rs, rt);
1975}
1976
1977void MipsAssembler::Beqz(Register rt, MipsLabel* label) {
1978 Bcond(label, kCondEQZ, rt);
1979}
1980
1981void MipsAssembler::Bnez(Register rt, MipsLabel* label) {
1982 Bcond(label, kCondNEZ, rt);
1983}
1984
1985void MipsAssembler::Bltz(Register rt, MipsLabel* label) {
1986 Bcond(label, kCondLTZ, rt);
1987}
1988
1989void MipsAssembler::Bgez(Register rt, MipsLabel* label) {
1990 Bcond(label, kCondGEZ, rt);
1991}
1992
1993void MipsAssembler::Blez(Register rt, MipsLabel* label) {
1994 Bcond(label, kCondLEZ, rt);
1995}
1996
1997void MipsAssembler::Bgtz(Register rt, MipsLabel* label) {
1998 Bcond(label, kCondGTZ, rt);
1999}
2000
2001void MipsAssembler::Blt(Register rs, Register rt, MipsLabel* label) {
2002 if (IsR6()) {
2003 Bcond(label, kCondLT, rs, rt);
2004 } else if (!Branch::IsNop(kCondLT, rs, rt)) {
2005 // Synthesize the instruction (not available on R2).
2006 Slt(AT, rs, rt);
2007 Bnez(AT, label);
2008 }
2009}
2010
2011void MipsAssembler::Bge(Register rs, Register rt, MipsLabel* label) {
2012 if (IsR6()) {
2013 Bcond(label, kCondGE, rs, rt);
2014 } else if (Branch::IsUncond(kCondGE, rs, rt)) {
2015 B(label);
2016 } else {
2017 // Synthesize the instruction (not available on R2).
2018 Slt(AT, rs, rt);
2019 Beqz(AT, label);
2020 }
2021}
2022
2023void MipsAssembler::Bltu(Register rs, Register rt, MipsLabel* label) {
2024 if (IsR6()) {
2025 Bcond(label, kCondLTU, rs, rt);
2026 } else if (!Branch::IsNop(kCondLTU, rs, rt)) {
2027 // Synthesize the instruction (not available on R2).
2028 Sltu(AT, rs, rt);
2029 Bnez(AT, label);
2030 }
2031}
2032
2033void MipsAssembler::Bgeu(Register rs, Register rt, MipsLabel* label) {
2034 if (IsR6()) {
2035 Bcond(label, kCondGEU, rs, rt);
2036 } else if (Branch::IsUncond(kCondGEU, rs, rt)) {
2037 B(label);
2038 } else {
2039 // Synthesize the instruction (not available on R2).
2040 Sltu(AT, rs, rt);
2041 Beqz(AT, label);
jeffhao7fbee072012-08-24 17:56:54 -07002042 }
2043}
2044
Alexey Frunzecd7b0ee2015-12-03 16:46:38 -08002045void MipsAssembler::Bc1f(int cc, MipsLabel* label) {
2046 CHECK(IsUint<3>(cc)) << cc;
2047 Bcond(label, kCondF, static_cast<Register>(cc), ZERO);
2048}
2049
2050void MipsAssembler::Bc1t(int cc, MipsLabel* label) {
2051 CHECK(IsUint<3>(cc)) << cc;
2052 Bcond(label, kCondT, static_cast<Register>(cc), ZERO);
2053}
2054
2055void MipsAssembler::Bc1eqz(FRegister ft, MipsLabel* label) {
2056 Bcond(label, kCondF, static_cast<Register>(ft), ZERO);
2057}
2058
2059void MipsAssembler::Bc1nez(FRegister ft, MipsLabel* label) {
2060 Bcond(label, kCondT, static_cast<Register>(ft), ZERO);
2061}
2062
jeffhao7fbee072012-08-24 17:56:54 -07002063void MipsAssembler::LoadFromOffset(LoadOperandType type, Register reg, Register base,
2064 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002065 // IsInt<16> must be passed a signed value.
2066 if (!IsInt<16>(offset) ||
2067 (type == kLoadDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2068 LoadConst32(AT, offset);
2069 Addu(AT, AT, base);
2070 base = AT;
2071 offset = 0;
2072 }
2073
jeffhao7fbee072012-08-24 17:56:54 -07002074 switch (type) {
2075 case kLoadSignedByte:
2076 Lb(reg, base, offset);
2077 break;
2078 case kLoadUnsignedByte:
2079 Lbu(reg, base, offset);
2080 break;
2081 case kLoadSignedHalfword:
2082 Lh(reg, base, offset);
2083 break;
2084 case kLoadUnsignedHalfword:
2085 Lhu(reg, base, offset);
2086 break;
2087 case kLoadWord:
2088 Lw(reg, base, offset);
2089 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002090 case kLoadDoubleword:
2091 if (reg == base) {
2092 // This will clobber the base when loading the lower register. Since we have to load the
2093 // higher register as well, this will fail. Solution: reverse the order.
2094 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
2095 Lw(reg, base, offset);
2096 } else {
2097 Lw(reg, base, offset);
2098 Lw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
2099 }
jeffhao7fbee072012-08-24 17:56:54 -07002100 break;
2101 default:
2102 LOG(FATAL) << "UNREACHABLE";
2103 }
2104}
2105
2106void MipsAssembler::LoadSFromOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002107 if (!IsInt<16>(offset)) {
2108 LoadConst32(AT, offset);
2109 Addu(AT, AT, base);
2110 base = AT;
2111 offset = 0;
2112 }
2113
jeffhao7fbee072012-08-24 17:56:54 -07002114 Lwc1(reg, base, offset);
2115}
2116
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002117void MipsAssembler::LoadDFromOffset(FRegister reg, Register base, int32_t offset) {
2118 // IsInt<16> must be passed a signed value.
2119 if (!IsInt<16>(offset) ||
2120 (!IsAligned<kMipsDoublewordSize>(offset) &&
2121 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2122 LoadConst32(AT, offset);
2123 Addu(AT, AT, base);
2124 base = AT;
2125 offset = 0;
2126 }
2127
2128 if (offset & 0x7) {
2129 if (Is32BitFPU()) {
2130 Lwc1(reg, base, offset);
2131 Lwc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
2132 } else {
2133 // 64-bit FPU.
2134 Lwc1(reg, base, offset);
2135 Lw(T8, base, offset + kMipsWordSize);
2136 Mthc1(T8, reg);
2137 }
2138 } else {
2139 Ldc1(reg, base, offset);
2140 }
2141}
2142
2143void MipsAssembler::EmitLoad(ManagedRegister m_dst, Register src_register, int32_t src_offset,
2144 size_t size) {
2145 MipsManagedRegister dst = m_dst.AsMips();
2146 if (dst.IsNoRegister()) {
2147 CHECK_EQ(0u, size) << dst;
2148 } else if (dst.IsCoreRegister()) {
2149 CHECK_EQ(kMipsWordSize, size) << dst;
2150 LoadFromOffset(kLoadWord, dst.AsCoreRegister(), src_register, src_offset);
2151 } else if (dst.IsRegisterPair()) {
2152 CHECK_EQ(kMipsDoublewordSize, size) << dst;
2153 LoadFromOffset(kLoadDoubleword, dst.AsRegisterPairLow(), src_register, src_offset);
2154 } else if (dst.IsFRegister()) {
2155 if (size == kMipsWordSize) {
2156 LoadSFromOffset(dst.AsFRegister(), src_register, src_offset);
2157 } else {
2158 CHECK_EQ(kMipsDoublewordSize, size) << dst;
2159 LoadDFromOffset(dst.AsFRegister(), src_register, src_offset);
2160 }
2161 }
jeffhao7fbee072012-08-24 17:56:54 -07002162}
2163
2164void MipsAssembler::StoreToOffset(StoreOperandType type, Register reg, Register base,
2165 int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002166 // IsInt<16> must be passed a signed value.
2167 if (!IsInt<16>(offset) ||
2168 (type == kStoreDoubleword && !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2169 LoadConst32(AT, offset);
2170 Addu(AT, AT, base);
2171 base = AT;
2172 offset = 0;
2173 }
2174
jeffhao7fbee072012-08-24 17:56:54 -07002175 switch (type) {
2176 case kStoreByte:
2177 Sb(reg, base, offset);
2178 break;
2179 case kStoreHalfword:
2180 Sh(reg, base, offset);
2181 break;
2182 case kStoreWord:
2183 Sw(reg, base, offset);
2184 break;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002185 case kStoreDoubleword:
2186 CHECK_NE(reg, base);
2187 CHECK_NE(static_cast<Register>(reg + 1), base);
2188 Sw(reg, base, offset);
2189 Sw(static_cast<Register>(reg + 1), base, offset + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002190 break;
2191 default:
2192 LOG(FATAL) << "UNREACHABLE";
2193 }
2194}
2195
Goran Jakovljevicff734982015-08-24 12:58:55 +00002196void MipsAssembler::StoreSToOffset(FRegister reg, Register base, int32_t offset) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002197 if (!IsInt<16>(offset)) {
2198 LoadConst32(AT, offset);
2199 Addu(AT, AT, base);
2200 base = AT;
2201 offset = 0;
2202 }
2203
jeffhao7fbee072012-08-24 17:56:54 -07002204 Swc1(reg, base, offset);
2205}
2206
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002207void MipsAssembler::StoreDToOffset(FRegister reg, Register base, int32_t offset) {
2208 // IsInt<16> must be passed a signed value.
2209 if (!IsInt<16>(offset) ||
2210 (!IsAligned<kMipsDoublewordSize>(offset) &&
2211 !IsInt<16>(static_cast<int32_t>(offset + kMipsWordSize)))) {
2212 LoadConst32(AT, offset);
2213 Addu(AT, AT, base);
2214 base = AT;
2215 offset = 0;
2216 }
2217
2218 if (offset & 0x7) {
2219 if (Is32BitFPU()) {
2220 Swc1(reg, base, offset);
2221 Swc1(static_cast<FRegister>(reg + 1), base, offset + kMipsWordSize);
2222 } else {
2223 // 64-bit FPU.
2224 Mfhc1(T8, reg);
2225 Swc1(reg, base, offset);
2226 Sw(T8, base, offset + kMipsWordSize);
2227 }
2228 } else {
2229 Sdc1(reg, base, offset);
2230 }
jeffhao7fbee072012-08-24 17:56:54 -07002231}
2232
David Srbeckydd973932015-04-07 20:29:48 +01002233static dwarf::Reg DWARFReg(Register reg) {
2234 return dwarf::Reg::MipsCore(static_cast<int>(reg));
2235}
2236
Ian Rogers790a6b72014-04-01 10:36:00 -07002237constexpr size_t kFramePointerSize = 4;
2238
jeffhao7fbee072012-08-24 17:56:54 -07002239void MipsAssembler::BuildFrame(size_t frame_size, ManagedRegister method_reg,
2240 const std::vector<ManagedRegister>& callee_save_regs,
Dmitry Petrochenkofca82202014-03-21 11:21:37 +07002241 const ManagedRegisterEntrySpills& entry_spills) {
jeffhao7fbee072012-08-24 17:56:54 -07002242 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002243 DCHECK(!overwriting_);
jeffhao7fbee072012-08-24 17:56:54 -07002244
2245 // Increase frame to required size.
2246 IncreaseFrameSize(frame_size);
2247
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002248 // Push callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07002249 int stack_offset = frame_size - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002250 StoreToOffset(kStoreWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002251 cfi_.RelOffset(DWARFReg(RA), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07002252 for (int i = callee_save_regs.size() - 1; i >= 0; --i) {
Ian Rogers790a6b72014-04-01 10:36:00 -07002253 stack_offset -= kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002254 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
2255 StoreToOffset(kStoreWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002256 cfi_.RelOffset(DWARFReg(reg), stack_offset);
jeffhao7fbee072012-08-24 17:56:54 -07002257 }
2258
2259 // Write out Method*.
2260 StoreToOffset(kStoreWord, method_reg.AsMips().AsCoreRegister(), SP, 0);
2261
2262 // Write out entry spills.
Goran Jakovljevicff734982015-08-24 12:58:55 +00002263 int32_t offset = frame_size + kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002264 for (size_t i = 0; i < entry_spills.size(); ++i) {
Goran Jakovljevicff734982015-08-24 12:58:55 +00002265 MipsManagedRegister reg = entry_spills.at(i).AsMips();
2266 if (reg.IsNoRegister()) {
2267 ManagedRegisterSpill spill = entry_spills.at(i);
2268 offset += spill.getSize();
2269 } else if (reg.IsCoreRegister()) {
2270 StoreToOffset(kStoreWord, reg.AsCoreRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002271 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002272 } else if (reg.IsFRegister()) {
2273 StoreSToOffset(reg.AsFRegister(), SP, offset);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002274 offset += kMipsWordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002275 } else if (reg.IsDRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002276 StoreDToOffset(reg.AsOverlappingDRegisterLow(), SP, offset);
2277 offset += kMipsDoublewordSize;
Goran Jakovljevicff734982015-08-24 12:58:55 +00002278 }
jeffhao7fbee072012-08-24 17:56:54 -07002279 }
2280}
2281
2282void MipsAssembler::RemoveFrame(size_t frame_size,
2283 const std::vector<ManagedRegister>& callee_save_regs) {
2284 CHECK_ALIGNED(frame_size, kStackAlignment);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002285 DCHECK(!overwriting_);
David Srbeckydd973932015-04-07 20:29:48 +01002286 cfi_.RememberState();
jeffhao7fbee072012-08-24 17:56:54 -07002287
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002288 // Pop callee saves and return address.
Ian Rogers790a6b72014-04-01 10:36:00 -07002289 int stack_offset = frame_size - (callee_save_regs.size() * kFramePointerSize) - kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002290 for (size_t i = 0; i < callee_save_regs.size(); ++i) {
2291 Register reg = callee_save_regs.at(i).AsMips().AsCoreRegister();
2292 LoadFromOffset(kLoadWord, reg, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002293 cfi_.Restore(DWARFReg(reg));
Ian Rogers790a6b72014-04-01 10:36:00 -07002294 stack_offset += kFramePointerSize;
jeffhao7fbee072012-08-24 17:56:54 -07002295 }
2296 LoadFromOffset(kLoadWord, RA, SP, stack_offset);
David Srbeckydd973932015-04-07 20:29:48 +01002297 cfi_.Restore(DWARFReg(RA));
jeffhao7fbee072012-08-24 17:56:54 -07002298
2299 // Decrease frame to required size.
2300 DecreaseFrameSize(frame_size);
jeffhao07030602012-09-26 14:33:14 -07002301
2302 // Then jump to the return address.
2303 Jr(RA);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002304 Nop();
David Srbeckydd973932015-04-07 20:29:48 +01002305
2306 // The CFI should be restored for any code that follows the exit block.
2307 cfi_.RestoreState();
2308 cfi_.DefCFAOffset(frame_size);
jeffhao7fbee072012-08-24 17:56:54 -07002309}
2310
2311void MipsAssembler::IncreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002312 CHECK_ALIGNED(adjust, kFramePointerSize);
2313 Addiu32(SP, SP, -adjust);
David Srbeckydd973932015-04-07 20:29:48 +01002314 cfi_.AdjustCFAOffset(adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002315 if (overwriting_) {
2316 cfi_.OverrideDelayedPC(overwrite_location_);
2317 }
jeffhao7fbee072012-08-24 17:56:54 -07002318}
2319
2320void MipsAssembler::DecreaseFrameSize(size_t adjust) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002321 CHECK_ALIGNED(adjust, kFramePointerSize);
2322 Addiu32(SP, SP, adjust);
David Srbeckydd973932015-04-07 20:29:48 +01002323 cfi_.AdjustCFAOffset(-adjust);
Vladimir Marko10ef6942015-10-22 15:25:54 +01002324 if (overwriting_) {
2325 cfi_.OverrideDelayedPC(overwrite_location_);
2326 }
jeffhao7fbee072012-08-24 17:56:54 -07002327}
2328
2329void MipsAssembler::Store(FrameOffset dest, ManagedRegister msrc, size_t size) {
2330 MipsManagedRegister src = msrc.AsMips();
2331 if (src.IsNoRegister()) {
2332 CHECK_EQ(0u, size);
2333 } else if (src.IsCoreRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002334 CHECK_EQ(kMipsWordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07002335 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2336 } else if (src.IsRegisterPair()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002337 CHECK_EQ(kMipsDoublewordSize, size);
jeffhao7fbee072012-08-24 17:56:54 -07002338 StoreToOffset(kStoreWord, src.AsRegisterPairLow(), SP, dest.Int32Value());
2339 StoreToOffset(kStoreWord, src.AsRegisterPairHigh(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002340 SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002341 } else if (src.IsFRegister()) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002342 if (size == kMipsWordSize) {
2343 StoreSToOffset(src.AsFRegister(), SP, dest.Int32Value());
2344 } else {
2345 CHECK_EQ(kMipsDoublewordSize, size);
2346 StoreDToOffset(src.AsFRegister(), SP, dest.Int32Value());
2347 }
jeffhao7fbee072012-08-24 17:56:54 -07002348 }
2349}
2350
2351void MipsAssembler::StoreRef(FrameOffset dest, ManagedRegister msrc) {
2352 MipsManagedRegister src = msrc.AsMips();
2353 CHECK(src.IsCoreRegister());
2354 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2355}
2356
2357void MipsAssembler::StoreRawPtr(FrameOffset dest, ManagedRegister msrc) {
2358 MipsManagedRegister src = msrc.AsMips();
2359 CHECK(src.IsCoreRegister());
2360 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2361}
2362
2363void MipsAssembler::StoreImmediateToFrame(FrameOffset dest, uint32_t imm,
2364 ManagedRegister mscratch) {
2365 MipsManagedRegister scratch = mscratch.AsMips();
2366 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002367 LoadConst32(scratch.AsCoreRegister(), imm);
jeffhao7fbee072012-08-24 17:56:54 -07002368 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2369}
2370
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002371void MipsAssembler::StoreImmediateToThread32(ThreadOffset<kMipsWordSize> dest, uint32_t imm,
jeffhao7fbee072012-08-24 17:56:54 -07002372 ManagedRegister mscratch) {
2373 MipsManagedRegister scratch = mscratch.AsMips();
2374 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002375 // Is this function even referenced anywhere else in the code?
2376 LoadConst32(scratch.AsCoreRegister(), imm);
2377 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), S1, dest.Int32Value());
2378}
2379
2380void MipsAssembler::StoreStackOffsetToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2381 FrameOffset fr_offs,
2382 ManagedRegister mscratch) {
2383 MipsManagedRegister scratch = mscratch.AsMips();
2384 CHECK(scratch.IsCoreRegister()) << scratch;
2385 Addiu32(scratch.AsCoreRegister(), SP, fr_offs.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002386 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2387 S1, thr_offs.Int32Value());
2388}
2389
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002390void MipsAssembler::StoreStackPointerToThread32(ThreadOffset<kMipsWordSize> thr_offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002391 StoreToOffset(kStoreWord, SP, S1, thr_offs.Int32Value());
2392}
2393
2394void MipsAssembler::StoreSpanning(FrameOffset dest, ManagedRegister msrc,
2395 FrameOffset in_off, ManagedRegister mscratch) {
2396 MipsManagedRegister src = msrc.AsMips();
2397 MipsManagedRegister scratch = mscratch.AsMips();
2398 StoreToOffset(kStoreWord, src.AsCoreRegister(), SP, dest.Int32Value());
2399 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, in_off.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002400 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002401}
2402
2403void MipsAssembler::Load(ManagedRegister mdest, FrameOffset src, size_t size) {
2404 return EmitLoad(mdest, SP, src.Int32Value(), size);
2405}
2406
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002407void MipsAssembler::LoadFromThread32(ManagedRegister mdest,
2408 ThreadOffset<kMipsWordSize> src, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002409 return EmitLoad(mdest, S1, src.Int32Value(), size);
2410}
2411
2412void MipsAssembler::LoadRef(ManagedRegister mdest, FrameOffset src) {
2413 MipsManagedRegister dest = mdest.AsMips();
2414 CHECK(dest.IsCoreRegister());
2415 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), SP, src.Int32Value());
2416}
2417
Mathieu Chartiere401d142015-04-22 13:56:20 -07002418void MipsAssembler::LoadRef(ManagedRegister mdest, ManagedRegister base, MemberOffset offs,
Roland Levillain4d027112015-07-01 15:41:14 +01002419 bool unpoison_reference) {
jeffhao7fbee072012-08-24 17:56:54 -07002420 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002421 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002422 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2423 base.AsMips().AsCoreRegister(), offs.Int32Value());
Roland Levillain4d027112015-07-01 15:41:14 +01002424 if (kPoisonHeapReferences && unpoison_reference) {
Hiroshi Yamauchie63a7452014-02-27 14:44:36 -08002425 Subu(dest.AsCoreRegister(), ZERO, dest.AsCoreRegister());
2426 }
jeffhao7fbee072012-08-24 17:56:54 -07002427}
2428
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002429void MipsAssembler::LoadRawPtr(ManagedRegister mdest, ManagedRegister base, Offset offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002430 MipsManagedRegister dest = mdest.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002431 CHECK(dest.IsCoreRegister() && base.AsMips().IsCoreRegister());
jeffhao7fbee072012-08-24 17:56:54 -07002432 LoadFromOffset(kLoadWord, dest.AsCoreRegister(),
2433 base.AsMips().AsCoreRegister(), offs.Int32Value());
2434}
2435
Ian Rogersdd7624d2014-03-14 17:43:00 -07002436void MipsAssembler::LoadRawPtrFromThread32(ManagedRegister mdest,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002437 ThreadOffset<kMipsWordSize> offs) {
jeffhao7fbee072012-08-24 17:56:54 -07002438 MipsManagedRegister dest = mdest.AsMips();
2439 CHECK(dest.IsCoreRegister());
2440 LoadFromOffset(kLoadWord, dest.AsCoreRegister(), S1, offs.Int32Value());
2441}
2442
2443void MipsAssembler::SignExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2444 UNIMPLEMENTED(FATAL) << "no sign extension necessary for mips";
2445}
2446
2447void MipsAssembler::ZeroExtend(ManagedRegister /*mreg*/, size_t /*size*/) {
2448 UNIMPLEMENTED(FATAL) << "no zero extension necessary for mips";
2449}
2450
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002451void MipsAssembler::Move(ManagedRegister mdest, ManagedRegister msrc, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002452 MipsManagedRegister dest = mdest.AsMips();
2453 MipsManagedRegister src = msrc.AsMips();
2454 if (!dest.Equals(src)) {
2455 if (dest.IsCoreRegister()) {
2456 CHECK(src.IsCoreRegister()) << src;
2457 Move(dest.AsCoreRegister(), src.AsCoreRegister());
2458 } else if (dest.IsFRegister()) {
2459 CHECK(src.IsFRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002460 if (size == kMipsWordSize) {
2461 MovS(dest.AsFRegister(), src.AsFRegister());
2462 } else {
2463 CHECK_EQ(kMipsDoublewordSize, size);
2464 MovD(dest.AsFRegister(), src.AsFRegister());
2465 }
jeffhao7fbee072012-08-24 17:56:54 -07002466 } else if (dest.IsDRegister()) {
2467 CHECK(src.IsDRegister()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002468 MovD(dest.AsOverlappingDRegisterLow(), src.AsOverlappingDRegisterLow());
jeffhao7fbee072012-08-24 17:56:54 -07002469 } else {
2470 CHECK(dest.IsRegisterPair()) << dest;
2471 CHECK(src.IsRegisterPair()) << src;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002472 // Ensure that the first move doesn't clobber the input of the second.
jeffhao7fbee072012-08-24 17:56:54 -07002473 if (src.AsRegisterPairHigh() != dest.AsRegisterPairLow()) {
2474 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2475 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2476 } else {
2477 Move(dest.AsRegisterPairHigh(), src.AsRegisterPairHigh());
2478 Move(dest.AsRegisterPairLow(), src.AsRegisterPairLow());
2479 }
2480 }
2481 }
2482}
2483
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002484void MipsAssembler::CopyRef(FrameOffset dest, FrameOffset src, ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002485 MipsManagedRegister scratch = mscratch.AsMips();
2486 CHECK(scratch.IsCoreRegister()) << scratch;
2487 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2488 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
2489}
2490
Ian Rogersdd7624d2014-03-14 17:43:00 -07002491void MipsAssembler::CopyRawPtrFromThread32(FrameOffset fr_offs,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002492 ThreadOffset<kMipsWordSize> thr_offs,
2493 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002494 MipsManagedRegister scratch = mscratch.AsMips();
2495 CHECK(scratch.IsCoreRegister()) << scratch;
2496 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2497 S1, thr_offs.Int32Value());
2498 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2499 SP, fr_offs.Int32Value());
2500}
2501
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002502void MipsAssembler::CopyRawPtrToThread32(ThreadOffset<kMipsWordSize> thr_offs,
2503 FrameOffset fr_offs,
2504 ManagedRegister mscratch) {
jeffhao7fbee072012-08-24 17:56:54 -07002505 MipsManagedRegister scratch = mscratch.AsMips();
2506 CHECK(scratch.IsCoreRegister()) << scratch;
2507 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2508 SP, fr_offs.Int32Value());
2509 StoreToOffset(kStoreWord, scratch.AsCoreRegister(),
2510 S1, thr_offs.Int32Value());
2511}
2512
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002513void MipsAssembler::Copy(FrameOffset dest, FrameOffset src, ManagedRegister mscratch, size_t size) {
jeffhao7fbee072012-08-24 17:56:54 -07002514 MipsManagedRegister scratch = mscratch.AsMips();
2515 CHECK(scratch.IsCoreRegister()) << scratch;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002516 CHECK(size == kMipsWordSize || size == kMipsDoublewordSize) << size;
2517 if (size == kMipsWordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002518 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2519 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002520 } else if (size == kMipsDoublewordSize) {
jeffhao7fbee072012-08-24 17:56:54 -07002521 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value());
2522 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002523 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, src.Int32Value() + kMipsWordSize);
2524 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, dest.Int32Value() + kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002525 }
2526}
2527
2528void MipsAssembler::Copy(FrameOffset dest, ManagedRegister src_base, Offset src_offset,
2529 ManagedRegister mscratch, size_t size) {
2530 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002531 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002532 LoadFromOffset(kLoadWord, scratch, src_base.AsMips().AsCoreRegister(), src_offset.Int32Value());
2533 StoreToOffset(kStoreWord, scratch, SP, dest.Int32Value());
2534}
2535
2536void MipsAssembler::Copy(ManagedRegister dest_base, Offset dest_offset, FrameOffset src,
2537 ManagedRegister mscratch, size_t size) {
2538 Register scratch = mscratch.AsMips().AsCoreRegister();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002539 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002540 LoadFromOffset(kLoadWord, scratch, SP, src.Int32Value());
2541 StoreToOffset(kStoreWord, scratch, dest_base.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2542}
2543
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002544void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2545 FrameOffset src_base ATTRIBUTE_UNUSED,
2546 Offset src_offset ATTRIBUTE_UNUSED,
2547 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2548 size_t size ATTRIBUTE_UNUSED) {
2549 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002550}
2551
2552void MipsAssembler::Copy(ManagedRegister dest, Offset dest_offset,
2553 ManagedRegister src, Offset src_offset,
2554 ManagedRegister mscratch, size_t size) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002555 CHECK_EQ(size, kMipsWordSize);
jeffhao7fbee072012-08-24 17:56:54 -07002556 Register scratch = mscratch.AsMips().AsCoreRegister();
2557 LoadFromOffset(kLoadWord, scratch, src.AsMips().AsCoreRegister(), src_offset.Int32Value());
2558 StoreToOffset(kStoreWord, scratch, dest.AsMips().AsCoreRegister(), dest_offset.Int32Value());
2559}
2560
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002561void MipsAssembler::Copy(FrameOffset dest ATTRIBUTE_UNUSED,
2562 Offset dest_offset ATTRIBUTE_UNUSED,
2563 FrameOffset src ATTRIBUTE_UNUSED,
2564 Offset src_offset ATTRIBUTE_UNUSED,
2565 ManagedRegister mscratch ATTRIBUTE_UNUSED,
2566 size_t size ATTRIBUTE_UNUSED) {
2567 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002568}
2569
2570void MipsAssembler::MemoryBarrier(ManagedRegister) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002571 // TODO: sync?
2572 UNIMPLEMENTED(FATAL) << "no MIPS implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002573}
2574
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002575void MipsAssembler::CreateHandleScopeEntry(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002576 FrameOffset handle_scope_offset,
2577 ManagedRegister min_reg,
2578 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002579 MipsManagedRegister out_reg = mout_reg.AsMips();
2580 MipsManagedRegister in_reg = min_reg.AsMips();
2581 CHECK(in_reg.IsNoRegister() || in_reg.IsCoreRegister()) << in_reg;
2582 CHECK(out_reg.IsCoreRegister()) << out_reg;
2583 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002584 MipsLabel null_arg;
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002585 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2586 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002587 // E.g. out_reg = (handle == 0) ? 0 : (SP+handle_offset).
jeffhao7fbee072012-08-24 17:56:54 -07002588 if (in_reg.IsNoRegister()) {
2589 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002590 SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002591 in_reg = out_reg;
2592 }
2593 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002594 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002595 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002596 Beqz(in_reg.AsCoreRegister(), &null_arg);
2597 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2598 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002599 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002600 Addiu32(out_reg.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002601 }
2602}
2603
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002604void MipsAssembler::CreateHandleScopeEntry(FrameOffset out_off,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002605 FrameOffset handle_scope_offset,
2606 ManagedRegister mscratch,
2607 bool null_allowed) {
jeffhao7fbee072012-08-24 17:56:54 -07002608 MipsManagedRegister scratch = mscratch.AsMips();
2609 CHECK(scratch.IsCoreRegister()) << scratch;
2610 if (null_allowed) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002611 MipsLabel null_arg;
2612 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002613 // Null values get a handle scope entry value of 0. Otherwise, the handle scope entry is
2614 // the address in the handle scope holding the reference.
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002615 // E.g. scratch = (scratch == 0) ? 0 : (SP+handle_scope_offset).
2616 Beqz(scratch.AsCoreRegister(), &null_arg);
2617 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
2618 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002619 } else {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002620 Addiu32(scratch.AsCoreRegister(), SP, handle_scope_offset.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002621 }
2622 StoreToOffset(kStoreWord, scratch.AsCoreRegister(), SP, out_off.Int32Value());
2623}
2624
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07002625// Given a handle scope entry, load the associated reference.
2626void MipsAssembler::LoadReferenceFromHandleScope(ManagedRegister mout_reg,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002627 ManagedRegister min_reg) {
jeffhao7fbee072012-08-24 17:56:54 -07002628 MipsManagedRegister out_reg = mout_reg.AsMips();
2629 MipsManagedRegister in_reg = min_reg.AsMips();
2630 CHECK(out_reg.IsCoreRegister()) << out_reg;
2631 CHECK(in_reg.IsCoreRegister()) << in_reg;
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002632 MipsLabel null_arg;
jeffhao7fbee072012-08-24 17:56:54 -07002633 if (!out_reg.Equals(in_reg)) {
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002634 LoadConst32(out_reg.AsCoreRegister(), 0);
jeffhao7fbee072012-08-24 17:56:54 -07002635 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002636 Beqz(in_reg.AsCoreRegister(), &null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002637 LoadFromOffset(kLoadWord, out_reg.AsCoreRegister(),
2638 in_reg.AsCoreRegister(), 0);
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002639 Bind(&null_arg);
jeffhao7fbee072012-08-24 17:56:54 -07002640}
2641
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002642void MipsAssembler::VerifyObject(ManagedRegister src ATTRIBUTE_UNUSED,
2643 bool could_be_null ATTRIBUTE_UNUSED) {
2644 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002645}
2646
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002647void MipsAssembler::VerifyObject(FrameOffset src ATTRIBUTE_UNUSED,
2648 bool could_be_null ATTRIBUTE_UNUSED) {
2649 // TODO: not validating references.
jeffhao7fbee072012-08-24 17:56:54 -07002650}
2651
2652void MipsAssembler::Call(ManagedRegister mbase, Offset offset, ManagedRegister mscratch) {
2653 MipsManagedRegister base = mbase.AsMips();
2654 MipsManagedRegister scratch = mscratch.AsMips();
2655 CHECK(base.IsCoreRegister()) << base;
2656 CHECK(scratch.IsCoreRegister()) << scratch;
2657 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2658 base.AsCoreRegister(), offset.Int32Value());
2659 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002660 Nop();
2661 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002662}
2663
2664void MipsAssembler::Call(FrameOffset base, Offset offset, ManagedRegister mscratch) {
2665 MipsManagedRegister scratch = mscratch.AsMips();
2666 CHECK(scratch.IsCoreRegister()) << scratch;
2667 // Call *(*(SP + base) + offset)
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002668 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(), SP, base.Int32Value());
jeffhao7fbee072012-08-24 17:56:54 -07002669 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
2670 scratch.AsCoreRegister(), offset.Int32Value());
2671 Jalr(scratch.AsCoreRegister());
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002672 Nop();
2673 // TODO: place reference map on call.
jeffhao7fbee072012-08-24 17:56:54 -07002674}
2675
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002676void MipsAssembler::CallFromThread32(ThreadOffset<kMipsWordSize> offset ATTRIBUTE_UNUSED,
2677 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
Ian Rogers468532e2013-08-05 10:56:33 -07002678 UNIMPLEMENTED(FATAL) << "no mips implementation";
jeffhao7fbee072012-08-24 17:56:54 -07002679}
2680
2681void MipsAssembler::GetCurrentThread(ManagedRegister tr) {
2682 Move(tr.AsMips().AsCoreRegister(), S1);
2683}
2684
2685void MipsAssembler::GetCurrentThread(FrameOffset offset,
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002686 ManagedRegister mscratch ATTRIBUTE_UNUSED) {
jeffhao7fbee072012-08-24 17:56:54 -07002687 StoreToOffset(kStoreWord, S1, SP, offset.Int32Value());
2688}
2689
jeffhao7fbee072012-08-24 17:56:54 -07002690void MipsAssembler::ExceptionPoll(ManagedRegister mscratch, size_t stack_adjust) {
2691 MipsManagedRegister scratch = mscratch.AsMips();
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002692 exception_blocks_.emplace_back(scratch, stack_adjust);
jeffhao7fbee072012-08-24 17:56:54 -07002693 LoadFromOffset(kLoadWord, scratch.AsCoreRegister(),
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002694 S1, Thread::ExceptionOffset<kMipsWordSize>().Int32Value());
2695 // TODO: on MIPS32R6 prefer Bnezc(scratch.AsCoreRegister(), slow.Entry());
2696 // as the NAL instruction (occurring in long R2 branches) may become deprecated.
2697 // For now use common for R2 and R6 instructions as this code must execute on both.
2698 Bnez(scratch.AsCoreRegister(), exception_blocks_.back().Entry());
jeffhao7fbee072012-08-24 17:56:54 -07002699}
2700
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002701void MipsAssembler::EmitExceptionPoll(MipsExceptionSlowPath* exception) {
2702 Bind(exception->Entry());
2703 if (exception->stack_adjust_ != 0) { // Fix up the frame.
2704 DecreaseFrameSize(exception->stack_adjust_);
jeffhao7fbee072012-08-24 17:56:54 -07002705 }
Goran Jakovljevic8c434dc2015-08-26 14:39:44 +02002706 // Pass exception object as argument.
2707 // Don't care about preserving A0 as this call won't return.
2708 CheckEntrypointTypes<kQuickDeliverException, void, mirror::Object*>();
2709 Move(A0, exception->scratch_.AsCoreRegister());
2710 // Set up call to Thread::Current()->pDeliverException.
2711 LoadFromOffset(kLoadWord, T9, S1,
2712 QUICK_ENTRYPOINT_OFFSET(kMipsWordSize, pDeliverException).Int32Value());
2713 Jr(T9);
2714 Nop();
2715
2716 // Call never returns.
2717 Break();
jeffhao7fbee072012-08-24 17:56:54 -07002718}
2719
2720} // namespace mips
2721} // namespace art