35 #include "src/mips/assembler-mips.h" 37 #if V8_TARGET_ARCH_MIPS 39 #include "src/base/bits.h" 40 #include "src/base/cpu.h" 41 #include "src/code-stubs.h" 42 #include "src/deoptimizer.h" 43 #include "src/mips/assembler-mips-inl.h" 44 #include "src/string-constants.h" 53 static unsigned CpuFeaturesImpliedByCompiler() {
55 #ifdef CAN_USE_FPU_INSTRUCTIONS 57 #endif // def CAN_USE_FPU_INSTRUCTIONS 62 #if defined(__mips__) && defined(__mips_hard_float) && __mips_hard_float != 0 70 void CpuFeatures::ProbeImpl(
bool cross_compile) {
71 supported_ |= CpuFeaturesImpliedByCompiler();
74 if (cross_compile)
return;
80 supported_ |= 1u << FPU;
81 #if defined(_MIPS_ARCH_MIPS32R6) 83 supported_ |= 1u << FP64FPU;
84 #if defined(_MIPS_MSA) 85 supported_ |= 1u << MIPS_SIMD;
88 #if defined(FPU_MODE_FP64) 89 supported_ |= 1u << FP64FPU;
94 if (cpu.has_fpu()) supported_ |= 1u << FPU;
95 #if defined(FPU_MODE_FPXX) 96 if (cpu.is_fp64_mode()) supported_ |= 1u << FP64FPU;
97 #elif defined(FPU_MODE_FP64) 98 supported_ |= 1u << FP64FPU;
99 #if defined(_MIPS_ARCH_MIPS32R6) 100 #if defined(_MIPS_MSA) 101 supported_ |= 1u << MIPS_SIMD;
103 if (cpu.has_msa()) supported_ |= 1u << MIPS_SIMD;
107 #if defined(_MIPS_ARCH_MIPS32RX) 108 if (cpu.architecture() == 6) {
109 supported_ |= 1u << MIPSr6;
110 }
else if (cpu.architecture() == 2) {
111 supported_ |= 1u << MIPSr1;
112 supported_ |= 1u << MIPSr2;
114 supported_ |= 1u << MIPSr1;
121 void CpuFeatures::PrintTarget() { }
122 void CpuFeatures::PrintFeatures() { }
125 int ToNumber(Register reg) {
126 DCHECK(reg.is_valid());
127 const int kNumbers[] = {
161 return kNumbers[reg.code()];
165 Register ToRegister(
int num) {
166 DCHECK(num >= 0 && num < kNumRegisters);
167 const Register kRegisters[] = {
172 t0, t1, t2, t3, t4, t5, t6, t7,
173 s0, s1, s2, s3, s4, s5, s6, s7,
181 return kRegisters[num];
188 const int RelocInfo::kApplyMask =
189 RelocInfo::ModeMask(RelocInfo::INTERNAL_REFERENCE) |
190 RelocInfo::ModeMask(RelocInfo::INTERNAL_REFERENCE_ENCODED) |
191 RelocInfo::ModeMask(RelocInfo::RELATIVE_CODE_TARGET);
193 bool RelocInfo::IsCodedSpecially() {
201 bool RelocInfo::IsInConstantPool() {
205 int RelocInfo::GetDeoptimizationId(Isolate* isolate, DeoptimizeKind kind) {
206 DCHECK(IsRuntimeEntry(rmode_));
207 return Deoptimizer::GetDeoptimizationId(isolate, target_address(), kind);
210 uint32_t RelocInfo::wasm_call_tag()
const {
211 DCHECK(rmode_ == WASM_CALL || rmode_ == WASM_STUB_CALL);
213 Assembler::target_address_at(pc_, constant_pool_));
220 Operand::Operand(Handle<HeapObject> handle)
221 : rm_(no_reg), rmode_(RelocInfo::EMBEDDED_OBJECT) {
222 value_.immediate =
static_cast<intptr_t
>(handle.address());
225 Operand Operand::EmbeddedNumber(
double value) {
227 if (DoubleToSmiInteger(value, &smi))
return Operand(Smi::FromInt(smi));
228 Operand result(0, RelocInfo::EMBEDDED_OBJECT);
229 result.is_heap_object_request_ =
true;
230 result.value_.heap_object_request = HeapObjectRequest(value);
234 Operand Operand::EmbeddedCode(CodeStub* stub) {
235 Operand result(0, RelocInfo::CODE_TARGET);
236 result.is_heap_object_request_ =
true;
237 result.value_.heap_object_request = HeapObjectRequest(stub);
241 Operand Operand::EmbeddedStringConstant(
const StringConstantBase* str) {
242 Operand result(0, RelocInfo::EMBEDDED_OBJECT);
243 result.is_heap_object_request_ =
true;
244 result.value_.heap_object_request = HeapObjectRequest(str);
248 MemOperand::MemOperand(Register rm, int32_t offset) : Operand(rm) {
253 MemOperand::MemOperand(Register rm, int32_t unit, int32_t multiplier,
254 OffsetAddend offset_addend) : Operand(rm) {
255 offset_ = unit * multiplier + offset_addend;
258 void Assembler::AllocateAndInstallRequestedHeapObjects(Isolate* isolate) {
259 DCHECK_IMPLIES(isolate ==
nullptr, heap_object_requests_.empty());
260 for (
auto& request : heap_object_requests_) {
261 Handle<HeapObject> object;
262 switch (request.kind()) {
263 case HeapObjectRequest::kHeapNumber:
265 isolate->factory()->NewHeapNumber(request.heap_number(), TENURED);
267 case HeapObjectRequest::kCodeStub:
268 request.code_stub()->set_isolate(isolate);
269 object = request.code_stub()->GetCode();
271 case HeapObjectRequest::kStringConstant:
272 const StringConstantBase* str = request.string();
274 object = str->AllocateStringConstant(isolate);
277 Address pc =
reinterpret_cast<Address
>(buffer_) + request.offset();
278 set_target_value_at(pc, reinterpret_cast<uint32_t>(
object.location()));
285 static const int kNegOffset = 0x00008000;
288 const Instr kPopInstruction = ADDIU | (sp.code() << kRsShift) |
289 (sp.code() << kRtShift) |
290 (kPointerSize & kImm16Mask);
292 const Instr kPushInstruction = ADDIU | (sp.code() << kRsShift) |
293 (sp.code() << kRtShift) |
294 (-kPointerSize & kImm16Mask);
296 const Instr kPushRegPattern =
297 SW | (sp.code() << kRsShift) | (0 & kImm16Mask);
299 const Instr kPopRegPattern =
300 LW | (sp.code() << kRsShift) | (0 & kImm16Mask);
302 const Instr kLwRegFpOffsetPattern =
303 LW | (fp.code() << kRsShift) | (0 & kImm16Mask);
305 const Instr kSwRegFpOffsetPattern =
306 SW | (fp.code() << kRsShift) | (0 & kImm16Mask);
308 const Instr kLwRegFpNegOffsetPattern =
309 LW | (fp.code() << kRsShift) | (kNegOffset & kImm16Mask);
311 const Instr kSwRegFpNegOffsetPattern =
312 SW | (fp.code() << kRsShift) | (kNegOffset & kImm16Mask);
314 const Instr kRtMask = kRtFieldMask;
315 const Instr kLwSwInstrTypeMask = 0xFFE00000;
316 const Instr kLwSwInstrArgumentMask = ~kLwSwInstrTypeMask;
317 const Instr kLwSwOffsetMask = kImm16Mask;
319 Assembler::Assembler(
const AssemblerOptions& options,
void* buffer,
321 : AssemblerBase(options, buffer, buffer_size),
322 scratch_register_list_(at.bit()) {
323 reloc_info_writer.Reposition(buffer_ + buffer_size_, pc_);
325 last_trampoline_pool_end_ = 0;
326 no_trampoline_pool_before_ = 0;
327 trampoline_pool_blocked_nesting_ = 0;
330 next_buffer_check_ = FLAG_force_long_branches
331 ? kMaxInt : kMaxBranchOffset - kTrampolineSlotsSize * 16;
332 internal_trampoline_exception_ =
false;
335 trampoline_emitted_ = FLAG_force_long_branches;
336 unbound_labels_count_ = 0;
337 block_buffer_growth_ =
false;
340 void Assembler::GetCode(Isolate* isolate, CodeDesc* desc) {
341 EmitForbiddenSlotInstruction();
342 DCHECK(pc_ <= reloc_info_writer.pos());
344 AllocateAndInstallRequestedHeapObjects(isolate);
347 desc->buffer = buffer_;
348 desc->buffer_size = buffer_size_;
349 desc->instr_size = pc_offset();
350 desc->reloc_size = (buffer_ + buffer_size_) - reloc_info_writer.pos();
352 desc->constant_pool_size = 0;
353 desc->unwinding_info_size = 0;
354 desc->unwinding_info =
nullptr;
358 void Assembler::Align(
int m) {
359 DCHECK(m >= 4 && base::bits::IsPowerOfTwo(m));
360 EmitForbiddenSlotInstruction();
361 while ((pc_offset() & (m - 1)) != 0) {
367 void Assembler::CodeTargetAlign() {
374 Register Assembler::GetRtReg(Instr instr) {
375 return Register::from_code((instr & kRtFieldMask) >> kRtShift);
379 Register Assembler::GetRsReg(Instr instr) {
380 return Register::from_code((instr & kRsFieldMask) >> kRsShift);
384 Register Assembler::GetRdReg(Instr instr) {
385 return Register::from_code((instr & kRdFieldMask) >> kRdShift);
389 uint32_t Assembler::GetRt(Instr instr) {
390 return (instr & kRtFieldMask) >> kRtShift;
394 uint32_t Assembler::GetRtField(Instr instr) {
395 return instr & kRtFieldMask;
399 uint32_t Assembler::GetRs(Instr instr) {
400 return (instr & kRsFieldMask) >> kRsShift;
404 uint32_t Assembler::GetRsField(Instr instr) {
405 return instr & kRsFieldMask;
409 uint32_t Assembler::GetRd(Instr instr) {
410 return (instr & kRdFieldMask) >> kRdShift;
414 uint32_t Assembler::GetRdField(Instr instr) {
415 return instr & kRdFieldMask;
419 uint32_t Assembler::GetSa(Instr instr) {
420 return (instr & kSaFieldMask) >> kSaShift;
424 uint32_t Assembler::GetSaField(Instr instr) {
425 return instr & kSaFieldMask;
429 uint32_t Assembler::GetOpcodeField(Instr instr) {
430 return instr & kOpcodeMask;
434 uint32_t Assembler::GetFunction(Instr instr) {
435 return (instr & kFunctionFieldMask) >> kFunctionShift;
439 uint32_t Assembler::GetFunctionField(Instr instr) {
440 return instr & kFunctionFieldMask;
444 uint32_t Assembler::GetImmediate16(Instr instr) {
445 return instr & kImm16Mask;
449 uint32_t Assembler::GetLabelConst(Instr instr) {
450 return instr & ~kImm16Mask;
454 bool Assembler::IsPop(Instr instr) {
455 return (instr & ~kRtMask) == kPopRegPattern;
459 bool Assembler::IsPush(Instr instr) {
460 return (instr & ~kRtMask) == kPushRegPattern;
464 bool Assembler::IsSwRegFpOffset(Instr instr) {
465 return ((instr & kLwSwInstrTypeMask) == kSwRegFpOffsetPattern);
469 bool Assembler::IsLwRegFpOffset(Instr instr) {
470 return ((instr & kLwSwInstrTypeMask) == kLwRegFpOffsetPattern);
474 bool Assembler::IsSwRegFpNegOffset(Instr instr) {
475 return ((instr & (kLwSwInstrTypeMask | kNegOffset)) ==
476 kSwRegFpNegOffsetPattern);
480 bool Assembler::IsLwRegFpNegOffset(Instr instr) {
481 return ((instr & (kLwSwInstrTypeMask | kNegOffset)) ==
482 kLwRegFpNegOffsetPattern);
501 const int kEndOfChain = -4;
503 const int kEndOfJumpChain = 0;
505 bool Assembler::IsMsaBranch(Instr instr) {
506 uint32_t opcode = GetOpcodeField(instr);
507 uint32_t rs_field = GetRsField(instr);
508 if (opcode == COP1) {
529 bool Assembler::IsBranch(Instr instr) {
530 uint32_t opcode = GetOpcodeField(instr);
531 uint32_t rt_field = GetRtField(instr);
532 uint32_t rs_field = GetRsField(instr);
535 opcode == BEQ || opcode == BNE || opcode == BLEZ || opcode == BGTZ ||
536 opcode == BEQL || opcode == BNEL || opcode == BLEZL || opcode == BGTZL ||
537 (opcode == REGIMM && (rt_field == BLTZ || rt_field == BGEZ ||
538 rt_field == BLTZAL || rt_field == BGEZAL)) ||
539 (opcode == COP1 && rs_field == BC1) ||
540 (opcode == COP1 && rs_field == BC1EQZ) ||
541 (opcode == COP1 && rs_field == BC1NEZ) || IsMsaBranch(instr);
542 if (!isBranch && IsMipsArchVariant(kMips32r6)) {
545 isBranch |= opcode == POP10 || opcode == POP30 || opcode == BC ||
547 (opcode == POP66 && rs_field != 0) ||
548 (opcode == POP76 && rs_field != 0);
554 bool Assembler::IsBc(Instr instr) {
555 uint32_t opcode = GetOpcodeField(instr);
557 return opcode == BC || opcode == BALC;
560 bool Assembler::IsNal(Instr instr) {
561 uint32_t opcode = GetOpcodeField(instr);
562 uint32_t rt_field = GetRtField(instr);
563 uint32_t rs_field = GetRsField(instr);
564 return opcode == REGIMM && rt_field == BLTZAL && rs_field == 0;
567 bool Assembler::IsBzc(Instr instr) {
568 uint32_t opcode = GetOpcodeField(instr);
570 return (opcode == POP66 && GetRsField(instr) != 0) ||
571 (opcode == POP76 && GetRsField(instr) != 0);
575 bool Assembler::IsEmittedConstant(Instr instr) {
576 uint32_t label_constant = GetLabelConst(instr);
577 return label_constant == 0;
581 bool Assembler::IsBeq(Instr instr) {
582 return GetOpcodeField(instr) == BEQ;
586 bool Assembler::IsBne(Instr instr) {
587 return GetOpcodeField(instr) == BNE;
591 bool Assembler::IsBeqzc(Instr instr) {
592 uint32_t opcode = GetOpcodeField(instr);
593 return opcode == POP66 && GetRsField(instr) != 0;
597 bool Assembler::IsBnezc(Instr instr) {
598 uint32_t opcode = GetOpcodeField(instr);
599 return opcode == POP76 && GetRsField(instr) != 0;
603 bool Assembler::IsBeqc(Instr instr) {
604 uint32_t opcode = GetOpcodeField(instr);
607 return opcode == POP10 && rs != 0 && rs < rt;
611 bool Assembler::IsBnec(Instr instr) {
612 uint32_t opcode = GetOpcodeField(instr);
615 return opcode == POP30 && rs != 0 && rs < rt;
618 bool Assembler::IsJicOrJialc(Instr instr) {
619 uint32_t opcode = GetOpcodeField(instr);
621 return (opcode == POP66 || opcode == POP76) && rs == 0;
624 bool Assembler::IsJump(Instr instr) {
625 uint32_t opcode = GetOpcodeField(instr);
626 uint32_t rt_field = GetRtField(instr);
627 uint32_t rd_field = GetRdField(instr);
628 uint32_t function_field = GetFunctionField(instr);
630 return opcode == J || opcode == JAL ||
631 (opcode == SPECIAL && rt_field == 0 &&
632 ((function_field == JALR) || (rd_field == 0 && (function_field == JR))));
635 bool Assembler::IsJ(Instr instr) {
636 uint32_t opcode = GetOpcodeField(instr);
642 bool Assembler::IsJal(Instr instr) {
643 return GetOpcodeField(instr) == JAL;
647 bool Assembler::IsJr(Instr instr) {
648 if (!IsMipsArchVariant(kMips32r6)) {
649 return GetOpcodeField(instr) == SPECIAL && GetFunctionField(instr) == JR;
651 return GetOpcodeField(instr) == SPECIAL &&
652 GetRdField(instr) == 0 && GetFunctionField(instr) == JALR;
657 bool Assembler::IsJalr(Instr instr) {
658 return GetOpcodeField(instr) == SPECIAL &&
659 GetRdField(instr) != 0 && GetFunctionField(instr) == JALR;
663 bool Assembler::IsLui(Instr instr) {
664 uint32_t opcode = GetOpcodeField(instr);
666 return opcode == LUI;
670 bool Assembler::IsOri(Instr instr) {
671 uint32_t opcode = GetOpcodeField(instr);
673 return opcode == ORI;
676 bool Assembler::IsAddu(Instr instr, Register rd, Register rs, Register rt) {
677 uint32_t opcode = GetOpcodeField(instr);
681 uint32_t sa_field = GetSaField(instr);
685 uint32_t function_field = GetFunction(instr);
686 return opcode == SPECIAL && sa_field == 0 && function_field == ADDU &&
687 rd_reg == rd_field && rs_reg == rs_field && rt_reg == rt_field;
690 bool Assembler::IsMov(Instr instr, Register rd, Register rs) {
691 uint32_t opcode = GetOpcodeField(instr);
697 uint32_t function_field = GetFunctionField(instr);
699 bool res = opcode == SPECIAL && function_field == OR && rd_field == rd_reg &&
700 rs_field == rs_reg && rt_field == 0;
704 bool Assembler::IsNop(Instr instr,
unsigned int type) {
707 uint32_t opcode = GetOpcodeField(instr);
708 uint32_t function = GetFunctionField(instr);
718 Register nop_rt_reg = (type == 0) ? zero_reg : at;
719 bool ret = (opcode == SPECIAL &&
function == SLL &&
720 rd ==
static_cast<uint32_t>(ToNumber(zero_reg)) &&
721 rt == static_cast<uint32_t>(ToNumber(nop_rt_reg)) &&
728 int32_t Assembler::GetBranchOffset(Instr instr) {
729 DCHECK(IsBranch(instr));
730 return (static_cast<int16_t>(instr & kImm16Mask)) << 2;
734 bool Assembler::IsLw(Instr instr) {
735 return (static_cast<uint32_t>(instr & kOpcodeMask) == LW);
739 int16_t Assembler::GetLwOffset(Instr instr) {
741 return ((instr & kImm16Mask));
745 Instr Assembler::SetLwOffset(Instr instr, int16_t offset) {
749 Instr temp_instr = LW | (instr & kRsFieldMask) | (instr & kRtFieldMask)
750 | (offset & kImm16Mask);
756 bool Assembler::IsSw(Instr instr) {
757 return (static_cast<uint32_t>(instr & kOpcodeMask) == SW);
761 Instr Assembler::SetSwOffset(Instr instr, int16_t offset) {
763 return ((instr & ~kImm16Mask) | (offset & kImm16Mask));
767 bool Assembler::IsAddImmediate(Instr instr) {
768 return ((instr & kOpcodeMask) == ADDIU);
772 Instr Assembler::SetAddImmediateOffset(Instr instr, int16_t offset) {
773 DCHECK(IsAddImmediate(instr));
774 return ((instr & ~kImm16Mask) | (offset & kImm16Mask));
778 bool Assembler::IsAndImmediate(Instr instr) {
779 return GetOpcodeField(instr) == ANDI;
783 static Assembler::OffsetSize OffsetSizeInBits(Instr instr) {
784 if (IsMipsArchVariant(kMips32r6)) {
785 if (Assembler::IsBc(instr)) {
786 return Assembler::OffsetSize::kOffset26;
787 }
else if (Assembler::IsBzc(instr)) {
788 return Assembler::OffsetSize::kOffset21;
791 return Assembler::OffsetSize::kOffset16;
795 static inline int32_t AddBranchOffset(
int pos, Instr instr) {
796 int bits = OffsetSizeInBits(instr);
797 const int32_t mask = (1 << bits) - 1;
802 int32_t imm = ((instr & mask) << bits) >> (bits - 2);
804 if (imm == kEndOfChain) {
808 return pos + Assembler::kBranchPCOffset + imm;
812 uint32_t Assembler::CreateTargetAddress(Instr instr_lui, Instr instr_jic) {
813 DCHECK(IsLui(instr_lui) && IsJicOrJialc(instr_jic));
814 int16_t jic_offset = GetImmediate16(instr_jic);
815 int16_t lui_offset = GetImmediate16(instr_lui);
817 if (jic_offset < 0) {
818 lui_offset += kImm16Mask;
823 return lui_offset_u | jic_offset_u;
831 void Assembler::UnpackTargetAddress(
uint32_t address, int16_t& lui_offset,
832 int16_t& jic_offset) {
833 lui_offset = (address & kHiMask) >> kLuiShift;
834 jic_offset = address & kLoMask;
836 if (jic_offset < 0) {
837 lui_offset -= kImm16Mask;
841 void Assembler::UnpackTargetAddressUnsigned(
uint32_t address,
844 int16_t lui_offset16 = (address & kHiMask) >> kLuiShift;
845 int16_t jic_offset16 = address & kLoMask;
847 if (jic_offset16 < 0) {
848 lui_offset16 -= kImm16Mask;
850 lui_offset =
static_cast<uint32_t>(lui_offset16) & kImm16Mask;
851 jic_offset =
static_cast<uint32_t>(jic_offset16) & kImm16Mask;
854 void Assembler::PatchLuiOriImmediate(
int pc, int32_t imm, Instr instr_lui,
855 Address offset_lui, Instr instr_ori,
856 Address offset_ori) {
857 DCHECK(IsLui(instr_lui));
858 DCHECK(IsOri(instr_ori));
859 instr_at_put(static_cast<int>(pc + offset_lui),
860 instr_lui | ((imm >> kLuiShift) & kImm16Mask));
861 instr_at_put(static_cast<int>(pc + offset_ori),
862 instr_ori | (imm & kImm16Mask));
865 void Assembler::PatchLuiOriImmediate(Address pc, int32_t imm, Instr instr_lui,
866 Address offset_lui, Instr instr_ori,
867 Address offset_ori) {
868 DCHECK(IsLui(instr_lui));
869 DCHECK(IsOri(instr_ori));
870 instr_at_put(pc + offset_lui, instr_lui | ((imm >> kLuiShift) & kImm16Mask));
871 instr_at_put(pc + offset_ori, instr_ori | (imm & kImm16Mask));
874 int32_t Assembler::GetLuiOriImmediate(Instr instr_lui, Instr instr_ori) {
875 DCHECK(IsLui(instr_lui));
876 DCHECK(IsOri(instr_ori));
878 imm = (instr_lui &
static_cast<int32_t
>(kImm16Mask)) << kLuiShift;
879 imm |= (instr_ori &
static_cast<int32_t
>(kImm16Mask));
883 int Assembler::target_at(
int pos,
bool is_internal) {
884 Instr instr = instr_at(pos);
889 int32_t instr_address =
reinterpret_cast<int32_t
>(buffer_ + pos);
890 int delta =
static_cast<int>(instr_address - instr);
895 if ((instr & ~kImm16Mask) == 0) {
900 int32_t imm18 =((instr &
static_cast<int32_t
>(kImm16Mask)) << 16) >> 14;
901 return (imm18 + pos);
905 DCHECK(IsBranch(instr) || IsLui(instr) || IsMov(instr, t8, ra));
906 if (IsBranch(instr)) {
907 return AddBranchOffset(pos, instr);
908 }
else if (IsMov(instr, t8, ra)) {
910 Instr instr_lui = instr_at(pos + 2 * kInstrSize);
911 Instr instr_ori = instr_at(pos + 3 * kInstrSize);
912 imm32 = GetLuiOriImmediate(instr_lui, instr_ori);
913 if (imm32 == kEndOfJumpChain) {
917 return pos + Assembler::kLongBranchPCOffset + imm32;
919 DCHECK(IsLui(instr));
920 if (IsNal(instr_at(pos + kInstrSize))) {
922 Instr instr_lui = instr_at(pos + 0 * kInstrSize);
923 Instr instr_ori = instr_at(pos + 2 * kInstrSize);
924 imm32 = GetLuiOriImmediate(instr_lui, instr_ori);
925 if (imm32 == kEndOfJumpChain) {
929 return pos + Assembler::kLongBranchPCOffset + imm32;
931 Instr instr1 = instr_at(pos + 0 * kInstrSize);
932 Instr instr2 = instr_at(pos + 1 * kInstrSize);
933 DCHECK(IsOri(instr2) || IsJicOrJialc(instr2));
935 if (IsJicOrJialc(instr2)) {
936 imm = CreateTargetAddress(instr1, instr2);
938 imm = GetLuiOriImmediate(instr1, instr2);
941 if (imm == kEndOfJumpChain) {
945 uint32_t instr_address =
reinterpret_cast<int32_t
>(buffer_ + pos);
946 int32_t delta = instr_address - imm;
956 static inline Instr SetBranchOffset(int32_t pos, int32_t target_pos,
958 int32_t bits = OffsetSizeInBits(instr);
959 int32_t imm = target_pos - (pos + Assembler::kBranchPCOffset);
960 DCHECK_EQ(imm & 3, 0);
963 const int32_t mask = (1 << bits) - 1;
965 DCHECK(is_intn(imm, bits));
967 return instr | (imm & mask);
971 void Assembler::target_at_put(int32_t pos, int32_t target_pos,
973 Instr instr = instr_at(pos);
977 instr_at_put(pos, imm);
980 if ((instr & ~kImm16Mask) == 0) {
981 DCHECK(target_pos == kEndOfChain || target_pos >= 0);
984 instr_at_put(pos, target_pos + (Code::kHeaderSize - kHeapObjectTag));
988 DCHECK(IsBranch(instr) || IsLui(instr) || IsMov(instr, t8, ra));
989 if (IsBranch(instr)) {
990 instr = SetBranchOffset(pos, target_pos, instr);
991 instr_at_put(pos, instr);
992 }
else if (IsMov(instr, t8, ra)) {
993 Instr instr_lui = instr_at(pos + 2 * kInstrSize);
994 Instr instr_ori = instr_at(pos + 3 * kInstrSize);
995 DCHECK(IsLui(instr_lui));
996 DCHECK(IsOri(instr_ori));
998 int32_t imm_short = target_pos - (pos + Assembler::kBranchPCOffset);
1000 if (is_int16(imm_short)) {
1003 Instr instr_b = BEQ;
1004 instr_b = SetBranchOffset(pos, target_pos, instr_b);
1006 Instr instr_j = instr_at(pos + 5 * kInstrSize);
1007 Instr instr_branch_delay;
1009 if (IsJump(instr_j)) {
1011 instr_branch_delay = nopInstr;
1014 instr_branch_delay = instr_at(pos + 7 * kInstrSize);
1016 instr_at_put(pos + 0 * kInstrSize, instr_b);
1017 instr_at_put(pos + 1 * kInstrSize, instr_branch_delay);
1019 int32_t imm = target_pos - (pos + Assembler::kLongBranchPCOffset);
1020 DCHECK_EQ(imm & 3, 0);
1022 instr_lui &= ~kImm16Mask;
1023 instr_ori &= ~kImm16Mask;
1025 PatchLuiOriImmediate(pos, imm, instr_lui, 2 * kInstrSize, instr_ori,
1029 DCHECK(IsLui(instr));
1030 if (IsNal(instr_at(pos + kInstrSize))) {
1031 Instr instr_lui = instr_at(pos + 0 * kInstrSize);
1032 Instr instr_ori = instr_at(pos + 2 * kInstrSize);
1033 DCHECK(IsLui(instr_lui));
1034 DCHECK(IsOri(instr_ori));
1035 int32_t imm = target_pos - (pos + Assembler::kLongBranchPCOffset);
1036 DCHECK_EQ(imm & 3, 0);
1037 if (is_int16(imm + Assembler::kLongBranchPCOffset -
1038 Assembler::kBranchPCOffset)) {
1041 Instr instr_b = REGIMM | BGEZAL;
1042 instr_b = SetBranchOffset(pos, target_pos, instr_b);
1045 Instr instr_a = ADDIU | ra.code() << kRsShift | ra.code() << kRtShift |
1046 kOptimizedBranchAndLinkLongReturnOffset;
1048 instr_at_put(pos, instr_b);
1049 instr_at_put(pos + 1 * kInstrSize, instr_a);
1051 instr_lui &= ~kImm16Mask;
1052 instr_ori &= ~kImm16Mask;
1053 PatchLuiOriImmediate(pos, imm, instr_lui, 0 * kInstrSize, instr_ori,
1057 Instr instr1 = instr_at(pos + 0 * kInstrSize);
1058 Instr instr2 = instr_at(pos + 1 * kInstrSize);
1059 DCHECK(IsOri(instr2) || IsJicOrJialc(instr2));
1061 DCHECK_EQ(imm & 3, 0);
1062 DCHECK(IsLui(instr1) && (IsJicOrJialc(instr2) || IsOri(instr2)));
1063 instr1 &= ~kImm16Mask;
1064 instr2 &= ~kImm16Mask;
1066 if (IsJicOrJialc(instr2)) {
1067 uint32_t lui_offset_u, jic_offset_u;
1068 UnpackTargetAddressUnsigned(imm, lui_offset_u, jic_offset_u);
1069 instr_at_put(pos + 0 * kInstrSize, instr1 | lui_offset_u);
1070 instr_at_put(pos + 1 * kInstrSize, instr2 | jic_offset_u);
1072 PatchLuiOriImmediate(pos, imm, instr1, 0 * kInstrSize, instr2,
1079 void Assembler::print(
const Label* L) {
1080 if (L->is_unused()) {
1081 PrintF(
"unused label\n");
1082 }
else if (L->is_bound()) {
1083 PrintF(
"bound label to %d\n", L->pos());
1084 }
else if (L->is_linked()) {
1086 l.link_to(L->pos());
1087 PrintF(
"unbound label");
1088 while (l.is_linked()) {
1089 PrintF(
"@ %d ", l.pos());
1090 Instr instr = instr_at(l.pos());
1091 if ((instr & ~kImm16Mask) == 0) {
1094 PrintF(
"%d\n", instr);
1096 next(&l, is_internal_reference(&l));
1099 PrintF(
"label in inconsistent state (pos = %d)\n", L->pos_);
1104 void Assembler::bind_to(Label* L,
int pos) {
1105 DCHECK(0 <= pos && pos <= pc_offset());
1106 int32_t trampoline_pos = kInvalidSlotPos;
1107 bool is_internal =
false;
1108 if (L->is_linked() && !trampoline_emitted_) {
1109 unbound_labels_count_--;
1110 if (!is_internal_reference(L)) {
1111 next_buffer_check_ += kTrampolineSlotsSize;
1115 while (L->is_linked()) {
1116 int32_t fixup_pos = L->pos();
1117 int32_t dist = pos - fixup_pos;
1118 is_internal = is_internal_reference(L);
1119 next(L, is_internal);
1121 Instr instr = instr_at(fixup_pos);
1123 target_at_put(fixup_pos, pos, is_internal);
1125 if (IsBranch(instr)) {
1126 int branch_offset = BranchOffset(instr);
1127 if (dist > branch_offset) {
1128 if (trampoline_pos == kInvalidSlotPos) {
1129 trampoline_pos = get_trampoline_entry(fixup_pos);
1130 CHECK_NE(trampoline_pos, kInvalidSlotPos);
1132 CHECK((trampoline_pos - fixup_pos) <= branch_offset);
1133 target_at_put(fixup_pos, trampoline_pos,
false);
1134 fixup_pos = trampoline_pos;
1136 target_at_put(fixup_pos, pos,
false);
1138 target_at_put(fixup_pos, pos,
false);
1146 if (pos > last_bound_pos_)
1147 last_bound_pos_ = pos;
1151 void Assembler::bind(Label* L) {
1152 DCHECK(!L->is_bound());
1153 bind_to(L, pc_offset());
1157 void Assembler::next(Label* L,
bool is_internal) {
1158 DCHECK(L->is_linked());
1159 int link = target_at(L->pos(), is_internal);
1160 if (link == kEndOfChain) {
1169 bool Assembler::is_near(Label* L) {
1170 DCHECK(L->is_bound());
1171 return pc_offset() - L->pos() < kMaxBranchOffset - 4 * kInstrSize;
1175 bool Assembler::is_near(Label* L, OffsetSize bits) {
1176 if (L ==
nullptr || !L->is_bound())
return true;
1177 return pc_offset() - L->pos() < (1 << (bits + 2 - 1)) - 1 - 5 * kInstrSize;
1181 bool Assembler::is_near_branch(Label* L) {
1182 DCHECK(L->is_bound());
1183 return IsMipsArchVariant(kMips32r6) ? is_near_r6(L) : is_near_pre_r6(L);
1187 int Assembler::BranchOffset(Instr instr) {
1189 int bits = OffsetSize::kOffset16;
1191 if (IsMipsArchVariant(kMips32r6)) {
1192 uint32_t opcode = GetOpcodeField(instr);
1197 bits = OffsetSize::kOffset26;
1203 if (GetRsField(instr) != 0) bits = OffsetSize::kOffset21;
1210 return (1 << (bits + 2 - 1)) - 1;
1218 bool Assembler::MustUseReg(RelocInfo::Mode rmode) {
1219 return !RelocInfo::IsNone(rmode);
1222 void Assembler::GenInstrRegister(Opcode opcode,
1227 SecondaryField func) {
1228 DCHECK(rd.is_valid() && rs.is_valid() && rt.is_valid() && is_uint5(sa));
1229 Instr instr = opcode | (rs.code() << kRsShift) | (rt.code() << kRtShift)
1230 | (rd.code() << kRdShift) | (sa << kSaShift) | func;
1235 void Assembler::GenInstrRegister(Opcode opcode,
1240 SecondaryField func) {
1241 DCHECK(rs.is_valid() && rt.is_valid() && is_uint5(msb) && is_uint5(lsb));
1242 Instr instr = opcode | (rs.code() << kRsShift) | (rt.code() << kRtShift)
1243 | (msb << kRdShift) | (lsb << kSaShift) | func;
1248 void Assembler::GenInstrRegister(Opcode opcode,
1253 SecondaryField func) {
1254 DCHECK(fd.is_valid() && fs.is_valid() && ft.is_valid());
1255 Instr instr = opcode | fmt | (ft.code() << kFtShift) | (fs.code() << kFsShift)
1256 | (fd.code() << kFdShift) | func;
1261 void Assembler::GenInstrRegister(Opcode opcode,
1266 SecondaryField func) {
1267 DCHECK(fd.is_valid() && fr.is_valid() && fs.is_valid() && ft.is_valid());
1268 Instr instr = opcode | (fr.code() << kFrShift) | (ft.code() << kFtShift)
1269 | (fs.code() << kFsShift) | (fd.code() << kFdShift) | func;
1274 void Assembler::GenInstrRegister(Opcode opcode,
1279 SecondaryField func) {
1280 DCHECK(fd.is_valid() && fs.is_valid() && rt.is_valid());
1281 Instr instr = opcode | fmt | (rt.code() << kRtShift)
1282 | (fs.code() << kFsShift) | (fd.code() << kFdShift) | func;
1287 void Assembler::GenInstrRegister(Opcode opcode,
1290 FPUControlRegister fs,
1291 SecondaryField func) {
1292 DCHECK(fs.is_valid() && rt.is_valid());
1294 opcode | fmt | (rt.code() << kRtShift) | (fs.code() << kFsShift) | func;
1301 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, Register rt,
1303 CompactBranchType is_compact_branch) {
1304 DCHECK(rs.is_valid() && rt.is_valid() && (is_int16(j) || is_uint16(j)));
1305 Instr instr = opcode | (rs.code() << kRsShift) | (rt.code() << kRtShift)
1307 emit(instr, is_compact_branch);
1310 void Assembler::GenInstrImmediate(Opcode opcode, Register base, Register rt,
1311 int32_t offset9,
int bit6,
1312 SecondaryField func) {
1313 DCHECK(base.is_valid() && rt.is_valid() && is_int9(offset9) &&
1315 Instr instr = opcode | (base.code() << kBaseShift) | (rt.code() << kRtShift) |
1316 ((offset9 << kImm9Shift) & kImm9Mask) | bit6 << kBit6Shift |
1321 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, SecondaryField SF,
1323 CompactBranchType is_compact_branch) {
1324 DCHECK(rs.is_valid() && (is_int16(j) || is_uint16(j)));
1325 Instr instr = opcode | (rs.code() << kRsShift) | SF | (j & kImm16Mask);
1326 emit(instr, is_compact_branch);
1330 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, FPURegister ft,
1332 CompactBranchType is_compact_branch) {
1333 DCHECK(rs.is_valid() && ft.is_valid() && (is_int16(j) || is_uint16(j)));
1334 Instr instr = opcode | (rs.code() << kRsShift) | (ft.code() << kFtShift)
1336 emit(instr, is_compact_branch);
1340 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, int32_t offset21,
1341 CompactBranchType is_compact_branch) {
1342 DCHECK(rs.is_valid() && (is_int21(offset21)));
1343 Instr instr = opcode | (rs.code() << kRsShift) | (offset21 & kImm21Mask);
1344 emit(instr, is_compact_branch);
1348 void Assembler::GenInstrImmediate(Opcode opcode, Register rs,
1350 DCHECK(rs.is_valid() && (is_uint21(offset21)));
1351 Instr instr = opcode | (rs.code() << kRsShift) | (offset21 & kImm21Mask);
1356 void Assembler::GenInstrImmediate(Opcode opcode, int32_t offset26,
1357 CompactBranchType is_compact_branch) {
1358 DCHECK(is_int26(offset26));
1359 Instr instr = opcode | (offset26 & kImm26Mask);
1360 emit(instr, is_compact_branch);
1364 void Assembler::GenInstrJump(Opcode opcode,
1366 BlockTrampolinePoolScope block_trampoline_pool(
this);
1367 DCHECK(is_uint26(address));
1368 Instr instr = opcode | address;
1370 BlockTrampolinePoolFor(1);
1374 void Assembler::GenInstrMsaI8(SecondaryField operation,
uint32_t imm8,
1375 MSARegister ws, MSARegister wd) {
1376 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1377 DCHECK(ws.is_valid() && wd.is_valid() && is_uint8(imm8));
1378 Instr instr = MSA | operation | ((imm8 & kImm8Mask) << kWtShift) |
1379 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1383 void Assembler::GenInstrMsaI5(SecondaryField operation, SecondaryField df,
1384 int32_t imm5, MSARegister ws, MSARegister wd) {
1385 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1386 DCHECK(ws.is_valid() && wd.is_valid());
1387 DCHECK((operation == MAXI_S) || (operation == MINI_S) ||
1388 (operation == CEQI) || (operation == CLTI_S) ||
1389 (operation == CLEI_S)
1392 Instr instr = MSA | operation | df | ((imm5 & kImm5Mask) << kWtShift) |
1393 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1397 void Assembler::GenInstrMsaBit(SecondaryField operation, SecondaryField df,
1398 uint32_t m, MSARegister ws, MSARegister wd) {
1399 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1400 DCHECK(ws.is_valid() && wd.is_valid() && is_valid_msa_df_m(df, m));
1401 Instr instr = MSA | operation | df | (m << kWtShift) |
1402 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1406 void Assembler::GenInstrMsaI10(SecondaryField operation, SecondaryField df,
1407 int32_t imm10, MSARegister wd) {
1408 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1409 DCHECK(wd.is_valid() && is_int10(imm10));
1410 Instr instr = MSA | operation | df | ((imm10 & kImm10Mask) << kWsShift) |
1411 (wd.code() << kWdShift);
1415 template <
typename RegType>
1416 void Assembler::GenInstrMsa3R(SecondaryField operation, SecondaryField df,
1417 RegType t, MSARegister ws, MSARegister wd) {
1418 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1419 DCHECK(t.is_valid() && ws.is_valid() && wd.is_valid());
1420 Instr instr = MSA | operation | df | (t.code() << kWtShift) |
1421 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1425 template <
typename DstType,
typename SrcType>
1426 void Assembler::GenInstrMsaElm(SecondaryField operation, SecondaryField df,
1427 uint32_t n, SrcType src, DstType dst) {
1428 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1429 DCHECK(src.is_valid() && dst.is_valid() && is_valid_msa_df_n(df, n));
1430 Instr instr = MSA | operation | df | (n << kWtShift) |
1431 (src.code() << kWsShift) | (dst.code() << kWdShift) |
1436 void Assembler::GenInstrMsa3RF(SecondaryField operation,
uint32_t df,
1437 MSARegister wt, MSARegister ws, MSARegister wd) {
1438 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1439 DCHECK(wt.is_valid() && ws.is_valid() && wd.is_valid());
1441 Instr instr = MSA | operation | (df << 21) | (wt.code() << kWtShift) |
1442 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1446 void Assembler::GenInstrMsaVec(SecondaryField operation, MSARegister wt,
1447 MSARegister ws, MSARegister wd) {
1448 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1449 DCHECK(wt.is_valid() && ws.is_valid() && wd.is_valid());
1450 Instr instr = MSA | operation | (wt.code() << kWtShift) |
1451 (ws.code() << kWsShift) | (wd.code() << kWdShift) |
1452 MSA_VEC_2R_2RF_MINOR;
1456 void Assembler::GenInstrMsaMI10(SecondaryField operation, int32_t s10,
1457 Register rs, MSARegister wd) {
1458 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1459 DCHECK(rs.is_valid() && wd.is_valid() && is_int10(s10));
1460 Instr instr = MSA | operation | ((s10 & kImm10Mask) << kWtShift) |
1461 (rs.code() << kWsShift) | (wd.code() << kWdShift);
1465 void Assembler::GenInstrMsa2R(SecondaryField operation, SecondaryField df,
1466 MSARegister ws, MSARegister wd) {
1467 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1468 DCHECK(ws.is_valid() && wd.is_valid());
1469 Instr instr = MSA | MSA_2R_FORMAT | operation | df | (ws.code() << kWsShift) |
1470 (wd.code() << kWdShift) | MSA_VEC_2R_2RF_MINOR;
1474 void Assembler::GenInstrMsa2RF(SecondaryField operation, SecondaryField df,
1475 MSARegister ws, MSARegister wd) {
1476 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1477 DCHECK(ws.is_valid() && wd.is_valid());
1478 Instr instr = MSA | MSA_2RF_FORMAT | operation | df |
1479 (ws.code() << kWsShift) | (wd.code() << kWdShift) |
1480 MSA_VEC_2R_2RF_MINOR;
1484 void Assembler::GenInstrMsaBranch(SecondaryField operation, MSARegister wt,
1486 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
1487 DCHECK(wt.is_valid() && is_int16(offset16));
1488 BlockTrampolinePoolScope block_trampoline_pool(
this);
1490 COP1 | operation | (wt.code() << kWtShift) | (offset16 & kImm16Mask);
1492 BlockTrampolinePoolFor(1);
1496 int32_t Assembler::get_trampoline_entry(int32_t pos) {
1497 int32_t trampoline_entry = kInvalidSlotPos;
1499 if (!internal_trampoline_exception_) {
1500 if (trampoline_.start() > pos) {
1501 trampoline_entry = trampoline_.take_slot();
1504 if (kInvalidSlotPos == trampoline_entry) {
1505 internal_trampoline_exception_ =
true;
1508 return trampoline_entry;
1512 uint32_t Assembler::jump_address(Label* L) {
1515 if (L->is_bound()) {
1516 target_pos = L->pos();
1518 if (L->is_linked()) {
1519 target_pos = L->pos();
1520 L->link_to(pc_offset());
1522 L->link_to(pc_offset());
1523 return kEndOfJumpChain;
1528 DCHECK_EQ(imm & 3, 0);
1533 uint32_t Assembler::branch_long_offset(Label* L) {
1536 if (L->is_bound()) {
1537 target_pos = L->pos();
1539 if (L->is_linked()) {
1540 target_pos = L->pos();
1541 L->link_to(pc_offset());
1543 L->link_to(pc_offset());
1544 return kEndOfJumpChain;
1548 DCHECK(is_int32(static_cast<int64_t>(target_pos) -
1549 static_cast<int64_t>(pc_offset() + kLongBranchPCOffset)));
1550 int32_t offset = target_pos - (pc_offset() + kLongBranchPCOffset);
1551 DCHECK_EQ(offset & 3, 0);
1556 int32_t Assembler::branch_offset_helper(Label* L, OffsetSize bits) {
1558 int32_t pad = IsPrevInstrCompactBranch() ? kInstrSize : 0;
1560 if (L->is_bound()) {
1561 target_pos = L->pos();
1563 if (L->is_linked()) {
1564 target_pos = L->pos();
1565 L->link_to(pc_offset() + pad);
1567 L->link_to(pc_offset() + pad);
1568 if (!trampoline_emitted_) {
1569 unbound_labels_count_++;
1570 next_buffer_check_ -= kTrampolineSlotsSize;
1576 int32_t offset = target_pos - (pc_offset() + kBranchPCOffset + pad);
1577 DCHECK(is_intn(offset, bits + 2));
1578 DCHECK_EQ(offset & 3, 0);
1584 void Assembler::label_at_put(Label* L,
int at_offset) {
1586 if (L->is_bound()) {
1587 target_pos = L->pos();
1588 instr_at_put(at_offset, target_pos + (Code::kHeaderSize - kHeapObjectTag));
1590 if (L->is_linked()) {
1591 target_pos = L->pos();
1592 int32_t imm18 = target_pos - at_offset;
1593 DCHECK_EQ(imm18 & 3, 0);
1594 int32_t imm16 = imm18 >> 2;
1595 DCHECK(is_int16(imm16));
1596 instr_at_put(at_offset, (imm16 & kImm16Mask));
1598 target_pos = kEndOfChain;
1599 instr_at_put(at_offset, 0);
1600 if (!trampoline_emitted_) {
1601 unbound_labels_count_++;
1602 next_buffer_check_ -= kTrampolineSlotsSize;
1605 L->link_to(at_offset);
1612 void Assembler::b(int16_t offset) {
1613 beq(zero_reg, zero_reg, offset);
1617 void Assembler::bal(int16_t offset) {
1618 bgezal(zero_reg, offset);
1622 void Assembler::bc(int32_t offset) {
1623 DCHECK(IsMipsArchVariant(kMips32r6));
1624 GenInstrImmediate(BC, offset, CompactBranchType::COMPACT_BRANCH);
1628 void Assembler::balc(int32_t offset) {
1629 DCHECK(IsMipsArchVariant(kMips32r6));
1630 GenInstrImmediate(BALC, offset, CompactBranchType::COMPACT_BRANCH);
1634 void Assembler::beq(Register rs, Register rt, int16_t offset) {
1635 BlockTrampolinePoolScope block_trampoline_pool(
this);
1636 GenInstrImmediate(BEQ, rs, rt, offset);
1637 BlockTrampolinePoolFor(1);
1641 void Assembler::bgez(Register rs, int16_t offset) {
1642 BlockTrampolinePoolScope block_trampoline_pool(
this);
1643 GenInstrImmediate(REGIMM, rs, BGEZ, offset);
1644 BlockTrampolinePoolFor(1);
1648 void Assembler::bgezc(Register rt, int16_t offset) {
1649 DCHECK(IsMipsArchVariant(kMips32r6));
1650 DCHECK(rt != zero_reg);
1651 GenInstrImmediate(BLEZL, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1655 void Assembler::bgeuc(Register rs, Register rt, int16_t offset) {
1656 DCHECK(IsMipsArchVariant(kMips32r6));
1657 DCHECK(rs != zero_reg);
1658 DCHECK(rt != zero_reg);
1659 DCHECK(rs.code() != rt.code());
1660 GenInstrImmediate(BLEZ, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1664 void Assembler::bgec(Register rs, Register rt, int16_t offset) {
1665 DCHECK(IsMipsArchVariant(kMips32r6));
1666 DCHECK(rs != zero_reg);
1667 DCHECK(rt != zero_reg);
1668 DCHECK(rs.code() != rt.code());
1669 GenInstrImmediate(BLEZL, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1673 void Assembler::bgezal(Register rs, int16_t offset) {
1674 DCHECK(!IsMipsArchVariant(kMips32r6) || rs == zero_reg);
1676 BlockTrampolinePoolScope block_trampoline_pool(
this);
1677 GenInstrImmediate(REGIMM, rs, BGEZAL, offset);
1678 BlockTrampolinePoolFor(1);
1682 void Assembler::bgtz(Register rs, int16_t offset) {
1683 BlockTrampolinePoolScope block_trampoline_pool(
this);
1684 GenInstrImmediate(BGTZ, rs, zero_reg, offset);
1685 BlockTrampolinePoolFor(1);
1689 void Assembler::bgtzc(Register rt, int16_t offset) {
1690 DCHECK(IsMipsArchVariant(kMips32r6));
1691 DCHECK(rt != zero_reg);
1692 GenInstrImmediate(BGTZL, zero_reg, rt, offset,
1693 CompactBranchType::COMPACT_BRANCH);
1697 void Assembler::blez(Register rs, int16_t offset) {
1698 BlockTrampolinePoolScope block_trampoline_pool(
this);
1699 GenInstrImmediate(BLEZ, rs, zero_reg, offset);
1700 BlockTrampolinePoolFor(1);
1704 void Assembler::blezc(Register rt, int16_t offset) {
1705 DCHECK(IsMipsArchVariant(kMips32r6));
1706 DCHECK(rt != zero_reg);
1707 GenInstrImmediate(BLEZL, zero_reg, rt, offset,
1708 CompactBranchType::COMPACT_BRANCH);
1712 void Assembler::bltzc(Register rt, int16_t offset) {
1713 DCHECK(IsMipsArchVariant(kMips32r6));
1714 DCHECK(rt != zero_reg);
1715 GenInstrImmediate(BGTZL, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1719 void Assembler::bltuc(Register rs, Register rt, int16_t offset) {
1720 DCHECK(IsMipsArchVariant(kMips32r6));
1721 DCHECK(rs != zero_reg);
1722 DCHECK(rt != zero_reg);
1723 DCHECK(rs.code() != rt.code());
1724 GenInstrImmediate(BGTZ, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1728 void Assembler::bltc(Register rs, Register rt, int16_t offset) {
1729 DCHECK(IsMipsArchVariant(kMips32r6));
1730 DCHECK(rs != zero_reg);
1731 DCHECK(rt != zero_reg);
1732 DCHECK(rs.code() != rt.code());
1733 GenInstrImmediate(BGTZL, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1737 void Assembler::bltz(Register rs, int16_t offset) {
1738 BlockTrampolinePoolScope block_trampoline_pool(
this);
1739 GenInstrImmediate(REGIMM, rs, BLTZ, offset);
1740 BlockTrampolinePoolFor(1);
1744 void Assembler::bltzal(Register rs, int16_t offset) {
1745 DCHECK(!IsMipsArchVariant(kMips32r6) || rs == zero_reg);
1747 BlockTrampolinePoolScope block_trampoline_pool(
this);
1748 GenInstrImmediate(REGIMM, rs, BLTZAL, offset);
1749 BlockTrampolinePoolFor(1);
1753 void Assembler::bne(Register rs, Register rt, int16_t offset) {
1754 BlockTrampolinePoolScope block_trampoline_pool(
this);
1755 GenInstrImmediate(BNE, rs, rt, offset);
1756 BlockTrampolinePoolFor(1);
1760 void Assembler::bovc(Register rs, Register rt, int16_t offset) {
1761 DCHECK(IsMipsArchVariant(kMips32r6));
1762 if (rs.code() >= rt.code()) {
1763 GenInstrImmediate(ADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1765 GenInstrImmediate(ADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1770 void Assembler::bnvc(Register rs, Register rt, int16_t offset) {
1771 DCHECK(IsMipsArchVariant(kMips32r6));
1772 if (rs.code() >= rt.code()) {
1773 GenInstrImmediate(DADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1775 GenInstrImmediate(DADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1780 void Assembler::blezalc(Register rt, int16_t offset) {
1781 DCHECK(IsMipsArchVariant(kMips32r6));
1782 DCHECK(rt != zero_reg);
1784 GenInstrImmediate(BLEZ, zero_reg, rt, offset,
1785 CompactBranchType::COMPACT_BRANCH);
1789 void Assembler::bgezalc(Register rt, int16_t offset) {
1790 DCHECK(IsMipsArchVariant(kMips32r6));
1791 DCHECK(rt != zero_reg);
1793 GenInstrImmediate(BLEZ, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1797 void Assembler::bgezall(Register rs, int16_t offset) {
1798 DCHECK(!IsMipsArchVariant(kMips32r6));
1799 DCHECK(rs != zero_reg);
1801 BlockTrampolinePoolScope block_trampoline_pool(
this);
1802 GenInstrImmediate(REGIMM, rs, BGEZALL, offset);
1803 BlockTrampolinePoolFor(1);
1807 void Assembler::bltzalc(Register rt, int16_t offset) {
1808 DCHECK(IsMipsArchVariant(kMips32r6));
1809 DCHECK(rt != zero_reg);
1811 GenInstrImmediate(BGTZ, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1815 void Assembler::bgtzalc(Register rt, int16_t offset) {
1816 DCHECK(IsMipsArchVariant(kMips32r6));
1817 DCHECK(rt != zero_reg);
1819 GenInstrImmediate(BGTZ, zero_reg, rt, offset,
1820 CompactBranchType::COMPACT_BRANCH);
1824 void Assembler::beqzalc(Register rt, int16_t offset) {
1825 DCHECK(IsMipsArchVariant(kMips32r6));
1826 DCHECK(rt != zero_reg);
1828 GenInstrImmediate(ADDI, zero_reg, rt, offset,
1829 CompactBranchType::COMPACT_BRANCH);
1833 void Assembler::bnezalc(Register rt, int16_t offset) {
1834 DCHECK(IsMipsArchVariant(kMips32r6));
1835 DCHECK(rt != zero_reg);
1837 GenInstrImmediate(DADDI, zero_reg, rt, offset,
1838 CompactBranchType::COMPACT_BRANCH);
1842 void Assembler::beqc(Register rs, Register rt, int16_t offset) {
1843 DCHECK(IsMipsArchVariant(kMips32r6));
1844 DCHECK(rs.code() != rt.code() && rs.code() != 0 && rt.code() != 0);
1845 if (rs.code() < rt.code()) {
1846 GenInstrImmediate(ADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1848 GenInstrImmediate(ADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1853 void Assembler::beqzc(Register rs, int32_t offset) {
1854 DCHECK(IsMipsArchVariant(kMips32r6));
1855 DCHECK(rs != zero_reg);
1856 GenInstrImmediate(POP66, rs, offset, CompactBranchType::COMPACT_BRANCH);
1860 void Assembler::bnec(Register rs, Register rt, int16_t offset) {
1861 DCHECK(IsMipsArchVariant(kMips32r6));
1862 DCHECK(rs.code() != rt.code() && rs.code() != 0 && rt.code() != 0);
1863 if (rs.code() < rt.code()) {
1864 GenInstrImmediate(DADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1866 GenInstrImmediate(DADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1871 void Assembler::bnezc(Register rs, int32_t offset) {
1872 DCHECK(IsMipsArchVariant(kMips32r6));
1873 DCHECK(rs != zero_reg);
1874 GenInstrImmediate(POP76, rs, offset, CompactBranchType::COMPACT_BRANCH);
1878 void Assembler::j(int32_t target) {
1882 bool in_range = ((ipc ^
static_cast<uint32_t>(target)) >>
1883 (kImm26Bits + kImmFieldShift)) == 0;
1884 DCHECK(in_range && ((target & 3) == 0));
1886 BlockTrampolinePoolScope block_trampoline_pool(
this);
1887 GenInstrJump(J, (target >> 2) & kImm26Mask);
1888 BlockTrampolinePoolFor(1);
1892 void Assembler::jr(Register rs) {
1893 if (!IsMipsArchVariant(kMips32r6)) {
1894 BlockTrampolinePoolScope block_trampoline_pool(
this);
1895 GenInstrRegister(SPECIAL, rs, zero_reg, zero_reg, 0, JR);
1896 BlockTrampolinePoolFor(1);
1903 void Assembler::jal(int32_t target) {
1907 bool in_range = ((ipc ^
static_cast<uint32_t>(target)) >>
1908 (kImm26Bits + kImmFieldShift)) == 0;
1909 DCHECK(in_range && ((target & 3) == 0));
1911 BlockTrampolinePoolScope block_trampoline_pool(
this);
1912 GenInstrJump(JAL, (target >> 2) & kImm26Mask);
1913 BlockTrampolinePoolFor(1);
1917 void Assembler::jalr(Register rs, Register rd) {
1918 DCHECK(rs.code() != rd.code());
1919 BlockTrampolinePoolScope block_trampoline_pool(
this);
1920 GenInstrRegister(SPECIAL, rs, zero_reg, rd, 0, JALR);
1921 BlockTrampolinePoolFor(1);
1925 void Assembler::jic(Register rt, int16_t offset) {
1926 DCHECK(IsMipsArchVariant(kMips32r6));
1927 GenInstrImmediate(POP66, zero_reg, rt, offset);
1931 void Assembler::jialc(Register rt, int16_t offset) {
1932 DCHECK(IsMipsArchVariant(kMips32r6));
1933 GenInstrImmediate(POP76, zero_reg, rt, offset);
1941 void Assembler::addu(Register rd, Register rs, Register rt) {
1942 GenInstrRegister(SPECIAL, rs, rt, rd, 0, ADDU);
1946 void Assembler::addiu(Register rd, Register rs, int32_t j) {
1947 GenInstrImmediate(ADDIU, rs, rd, j);
1951 void Assembler::subu(Register rd, Register rs, Register rt) {
1952 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SUBU);
1956 void Assembler::mul(Register rd, Register rs, Register rt) {
1957 if (!IsMipsArchVariant(kMips32r6)) {
1958 GenInstrRegister(SPECIAL2, rs, rt, rd, 0, MUL);
1960 GenInstrRegister(SPECIAL, rs, rt, rd, MUL_OP, MUL_MUH);
1965 void Assembler::mulu(Register rd, Register rs, Register rt) {
1966 DCHECK(IsMipsArchVariant(kMips32r6));
1967 GenInstrRegister(SPECIAL, rs, rt, rd, MUL_OP, MUL_MUH_U);
1971 void Assembler::muh(Register rd, Register rs, Register rt) {
1972 DCHECK(IsMipsArchVariant(kMips32r6));
1973 GenInstrRegister(SPECIAL, rs, rt, rd, MUH_OP, MUL_MUH);
1977 void Assembler::muhu(Register rd, Register rs, Register rt) {
1978 DCHECK(IsMipsArchVariant(kMips32r6));
1979 GenInstrRegister(SPECIAL, rs, rt, rd, MUH_OP, MUL_MUH_U);
1983 void Assembler::mod(Register rd, Register rs, Register rt) {
1984 DCHECK(IsMipsArchVariant(kMips32r6));
1985 GenInstrRegister(SPECIAL, rs, rt, rd, MOD_OP, DIV_MOD);
1989 void Assembler::modu(Register rd, Register rs, Register rt) {
1990 DCHECK(IsMipsArchVariant(kMips32r6));
1991 GenInstrRegister(SPECIAL, rs, rt, rd, MOD_OP, DIV_MOD_U);
1995 void Assembler::mult(Register rs, Register rt) {
1996 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, MULT);
2000 void Assembler::multu(Register rs, Register rt) {
2001 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, MULTU);
2005 void Assembler::div(Register rs, Register rt) {
2006 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DIV);
2010 void Assembler::div(Register rd, Register rs, Register rt) {
2011 DCHECK(IsMipsArchVariant(kMips32r6));
2012 GenInstrRegister(SPECIAL, rs, rt, rd, DIV_OP, DIV_MOD);
2016 void Assembler::divu(Register rs, Register rt) {
2017 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DIVU);
2021 void Assembler::divu(Register rd, Register rs, Register rt) {
2022 DCHECK(IsMipsArchVariant(kMips32r6));
2023 GenInstrRegister(SPECIAL, rs, rt, rd, DIV_OP, DIV_MOD_U);
2029 void Assembler::and_(Register rd, Register rs, Register rt) {
2030 GenInstrRegister(SPECIAL, rs, rt, rd, 0, AND);
2034 void Assembler::andi(Register rt, Register rs, int32_t j) {
2035 DCHECK(is_uint16(j));
2036 GenInstrImmediate(ANDI, rs, rt, j);
2040 void Assembler::or_(Register rd, Register rs, Register rt) {
2041 GenInstrRegister(SPECIAL, rs, rt, rd, 0, OR);
2045 void Assembler::ori(Register rt, Register rs, int32_t j) {
2046 DCHECK(is_uint16(j));
2047 GenInstrImmediate(ORI, rs, rt, j);
2051 void Assembler::xor_(Register rd, Register rs, Register rt) {
2052 GenInstrRegister(SPECIAL, rs, rt, rd, 0, XOR);
2056 void Assembler::xori(Register rt, Register rs, int32_t j) {
2057 DCHECK(is_uint16(j));
2058 GenInstrImmediate(XORI, rs, rt, j);
2062 void Assembler::nor(Register rd, Register rs, Register rt) {
2063 GenInstrRegister(SPECIAL, rs, rt, rd, 0, NOR);
2068 void Assembler::sll(Register rd,
2071 bool coming_from_nop) {
2075 DCHECK(coming_from_nop || !(rd == zero_reg && rt == zero_reg));
2076 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, SLL);
2080 void Assembler::sllv(Register rd, Register rt, Register rs) {
2081 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SLLV);
2085 void Assembler::srl(Register rd, Register rt, uint16_t sa) {
2086 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, SRL);
2090 void Assembler::srlv(Register rd, Register rt, Register rs) {
2091 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SRLV);
2095 void Assembler::sra(Register rd, Register rt, uint16_t sa) {
2096 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, SRA);
2100 void Assembler::srav(Register rd, Register rt, Register rs) {
2101 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SRAV);
2105 void Assembler::rotr(Register rd, Register rt, uint16_t sa) {
2107 DCHECK(rd.is_valid() && rt.is_valid() && is_uint5(sa));
2108 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2109 Instr instr = SPECIAL | (1 << kRsShift) | (rt.code() << kRtShift)
2110 | (rd.code() << kRdShift) | (sa << kSaShift) | SRL;
2115 void Assembler::rotrv(Register rd, Register rt, Register rs) {
2117 DCHECK(rd.is_valid() && rt.is_valid() && rs.is_valid());
2118 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2119 Instr instr = SPECIAL | (rs.code() << kRsShift) | (rt.code() << kRtShift)
2120 | (rd.code() << kRdShift) | (1 << kSaShift) | SRLV;
2125 void Assembler::lsa(Register rd, Register rt, Register rs, uint8_t sa) {
2126 DCHECK(rd.is_valid() && rt.is_valid() && rs.is_valid());
2128 DCHECK(IsMipsArchVariant(kMips32r6));
2129 Instr instr = SPECIAL | rs.code() << kRsShift | rt.code() << kRtShift |
2130 rd.code() << kRdShift | sa << kSaShift | LSA;
2137 void Assembler::AdjustBaseAndOffset(MemOperand& src,
2138 OffsetAccessType access_type,
2139 int second_access_add_to_offset) {
2150 bool doubleword_aligned = (src.offset() & (kDoubleSize - 1)) == 0;
2151 bool two_accesses =
static_cast<bool>(access_type) || !doubleword_aligned;
2152 DCHECK_LE(second_access_add_to_offset, 7);
2155 if (is_int16(src.offset()) &&
2156 (!two_accesses || is_int16(static_cast<int32_t>(
2157 src.offset() + second_access_add_to_offset)))) {
2162 UseScratchRegisterScope temps(
this);
2163 Register scratch = temps.Acquire();
2164 DCHECK(src.rm() != scratch);
2169 uint32_t misalignment = src.offset() & (kDoubleSize - 1);
2177 constexpr int32_t kMinOffsetForSimpleAdjustment =
2179 constexpr int32_t kMaxOffsetForSimpleAdjustment =
2180 2 * kMinOffsetForSimpleAdjustment;
2181 if (0 <= src.offset() && src.offset() <= kMaxOffsetForSimpleAdjustment) {
2182 addiu(at, src.rm(), kMinOffsetForSimpleAdjustment);
2183 src.offset_ -= kMinOffsetForSimpleAdjustment;
2184 }
else if (-kMaxOffsetForSimpleAdjustment <= src.offset() &&
2186 addiu(at, src.rm(), -kMinOffsetForSimpleAdjustment);
2187 src.offset_ += kMinOffsetForSimpleAdjustment;
2188 }
else if (IsMipsArchVariant(kMips32r6)) {
2198 int16_t offset_high =
static_cast<uint16_t
>(src.offset() >> 16);
2199 int16_t offset_low =
static_cast<uint16_t
>(src.offset());
2200 offset_high += (offset_low < 0)
2203 aui(scratch, src.rm(),
static_cast<uint16_t
>(offset_high));
2204 if (two_accesses && !is_int16(static_cast<int32_t>(
2205 offset_low + second_access_add_to_offset))) {
2208 addiu(scratch, scratch, kDoubleSize);
2209 offset_low -= kDoubleSize;
2211 src.offset_ = offset_low;
2218 constexpr int32_t kMinOffsetForMediumAdjustment =
2219 2 * kMinOffsetForSimpleAdjustment;
2220 constexpr int32_t kMaxOffsetForMediumAdjustment =
2221 3 * kMinOffsetForSimpleAdjustment;
2222 if (0 <= src.offset() && src.offset() <= kMaxOffsetForMediumAdjustment) {
2223 addiu(scratch, src.rm(), kMinOffsetForMediumAdjustment / 2);
2224 addiu(scratch, scratch, kMinOffsetForMediumAdjustment / 2);
2225 src.offset_ -= kMinOffsetForMediumAdjustment;
2226 }
else if (-kMaxOffsetForMediumAdjustment <= src.offset() &&
2228 addiu(scratch, src.rm(), -kMinOffsetForMediumAdjustment / 2);
2229 addiu(scratch, scratch, -kMinOffsetForMediumAdjustment / 2);
2230 src.offset_ += kMinOffsetForMediumAdjustment;
2234 int32_t loaded_offset = RoundDown(src.offset(), kDoubleSize);
2235 lui(scratch, (loaded_offset >> kLuiShift) & kImm16Mask);
2236 ori(scratch, scratch, loaded_offset & kImm16Mask);
2237 addu(scratch, scratch, src.rm());
2238 src.offset_ -= loaded_offset;
2243 DCHECK(is_int16(src.offset()));
2246 static_cast<int32_t>(src.offset() + second_access_add_to_offset)));
2248 DCHECK(misalignment == (src.offset() & (kDoubleSize - 1)));
2251 void Assembler::lb(Register rd,
const MemOperand& rs) {
2252 MemOperand source = rs;
2253 AdjustBaseAndOffset(source);
2254 GenInstrImmediate(LB, source.rm(), rd, source.offset());
2258 void Assembler::lbu(Register rd,
const MemOperand& rs) {
2259 MemOperand source = rs;
2260 AdjustBaseAndOffset(source);
2261 GenInstrImmediate(LBU, source.rm(), rd, source.offset());
2265 void Assembler::lh(Register rd,
const MemOperand& rs) {
2266 MemOperand source = rs;
2267 AdjustBaseAndOffset(source);
2268 GenInstrImmediate(LH, source.rm(), rd, source.offset());
2272 void Assembler::lhu(Register rd,
const MemOperand& rs) {
2273 MemOperand source = rs;
2274 AdjustBaseAndOffset(source);
2275 GenInstrImmediate(LHU, source.rm(), rd, source.offset());
2279 void Assembler::lw(Register rd,
const MemOperand& rs) {
2280 MemOperand source = rs;
2281 AdjustBaseAndOffset(source);
2282 GenInstrImmediate(LW, source.rm(), rd, source.offset());
2286 void Assembler::lwl(Register rd,
const MemOperand& rs) {
2287 DCHECK(is_int16(rs.offset_));
2288 DCHECK(IsMipsArchVariant(kLoongson) || IsMipsArchVariant(kMips32r1) ||
2289 IsMipsArchVariant(kMips32r2));
2290 GenInstrImmediate(LWL, rs.rm(), rd, rs.offset_);
2294 void Assembler::lwr(Register rd,
const MemOperand& rs) {
2295 DCHECK(is_int16(rs.offset_));
2296 DCHECK(IsMipsArchVariant(kLoongson) || IsMipsArchVariant(kMips32r1) ||
2297 IsMipsArchVariant(kMips32r2));
2298 GenInstrImmediate(LWR, rs.rm(), rd, rs.offset_);
2302 void Assembler::sb(Register rd,
const MemOperand& rs) {
2303 MemOperand source = rs;
2304 AdjustBaseAndOffset(source);
2305 GenInstrImmediate(SB, source.rm(), rd, source.offset());
2309 void Assembler::sh(Register rd,
const MemOperand& rs) {
2310 MemOperand source = rs;
2311 AdjustBaseAndOffset(source);
2312 GenInstrImmediate(SH, source.rm(), rd, source.offset());
2316 void Assembler::sw(Register rd,
const MemOperand& rs) {
2317 MemOperand source = rs;
2318 AdjustBaseAndOffset(source);
2319 GenInstrImmediate(SW, source.rm(), rd, source.offset());
2323 void Assembler::swl(Register rd,
const MemOperand& rs) {
2324 DCHECK(is_int16(rs.offset_));
2325 DCHECK(IsMipsArchVariant(kLoongson) || IsMipsArchVariant(kMips32r1) ||
2326 IsMipsArchVariant(kMips32r2));
2327 GenInstrImmediate(SWL, rs.rm(), rd, rs.offset_);
2331 void Assembler::swr(Register rd,
const MemOperand& rs) {
2332 DCHECK(is_int16(rs.offset_));
2333 DCHECK(IsMipsArchVariant(kLoongson) || IsMipsArchVariant(kMips32r1) ||
2334 IsMipsArchVariant(kMips32r2));
2335 GenInstrImmediate(SWR, rs.rm(), rd, rs.offset_);
2338 void Assembler::ll(Register rd,
const MemOperand& rs) {
2339 if (IsMipsArchVariant(kMips32r6)) {
2340 DCHECK(is_int9(rs.offset_));
2341 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 0, LL_R6);
2343 DCHECK(IsMipsArchVariant(kLoongson) || IsMipsArchVariant(kMips32r1) ||
2344 IsMipsArchVariant(kMips32r2));
2345 DCHECK(is_int16(rs.offset_));
2346 GenInstrImmediate(LL, rs.rm(), rd, rs.offset_);
2350 void Assembler::sc(Register rd,
const MemOperand& rs) {
2351 if (IsMipsArchVariant(kMips32r6)) {
2352 DCHECK(is_int9(rs.offset_));
2353 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 0, SC_R6);
2355 DCHECK(IsMipsArchVariant(kLoongson) || IsMipsArchVariant(kMips32r1) ||
2356 IsMipsArchVariant(kMips32r2));
2357 GenInstrImmediate(SC, rs.rm(), rd, rs.offset_);
2361 void Assembler::llx(Register rd,
const MemOperand& rs) {
2362 DCHECK(IsMipsArchVariant(kMips32r6));
2363 DCHECK(is_int9(rs.offset_));
2364 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 1, LL_R6);
2367 void Assembler::scx(Register rd,
const MemOperand& rs) {
2368 DCHECK(IsMipsArchVariant(kMips32r6));
2369 DCHECK(is_int9(rs.offset_));
2370 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 1, SC_R6);
2373 void Assembler::lui(Register rd, int32_t j) {
2374 DCHECK(is_uint16(j) || is_int16(j));
2375 GenInstrImmediate(LUI, zero_reg, rd, j);
2379 void Assembler::aui(Register rt, Register rs, int32_t j) {
2382 DCHECK(IsMipsArchVariant(kMips32r6));
2383 DCHECK(rs != zero_reg);
2384 DCHECK(is_uint16(j));
2385 GenInstrImmediate(LUI, rs, rt, j);
2390 void Assembler::addiupc(Register rs, int32_t imm19) {
2391 DCHECK(IsMipsArchVariant(kMips32r6));
2392 DCHECK(rs.is_valid() && is_int19(imm19));
2393 uint32_t imm21 = ADDIUPC << kImm19Bits | (imm19 & kImm19Mask);
2394 GenInstrImmediate(PCREL, rs, imm21);
2398 void Assembler::lwpc(Register rs, int32_t offset19) {
2399 DCHECK(IsMipsArchVariant(kMips32r6));
2400 DCHECK(rs.is_valid() && is_int19(offset19));
2401 uint32_t imm21 = LWPC << kImm19Bits | (offset19 & kImm19Mask);
2402 GenInstrImmediate(PCREL, rs, imm21);
2406 void Assembler::auipc(Register rs, int16_t imm16) {
2407 DCHECK(IsMipsArchVariant(kMips32r6));
2408 DCHECK(rs.is_valid());
2409 uint32_t imm21 = AUIPC << kImm16Bits | (imm16 & kImm16Mask);
2410 GenInstrImmediate(PCREL, rs, imm21);
2414 void Assembler::aluipc(Register rs, int16_t imm16) {
2415 DCHECK(IsMipsArchVariant(kMips32r6));
2416 DCHECK(rs.is_valid());
2417 uint32_t imm21 = ALUIPC << kImm16Bits | (imm16 & kImm16Mask);
2418 GenInstrImmediate(PCREL, rs, imm21);
2425 void Assembler::break_(
uint32_t code,
bool break_as_stop) {
2426 DCHECK_EQ(code & ~0xFFFFF, 0);
2430 DCHECK((break_as_stop &&
2431 code <= kMaxStopCode &&
2432 code > kMaxWatchpointCode) ||
2434 (code > kMaxStopCode ||
2435 code <= kMaxWatchpointCode)));
2436 Instr break_instr = SPECIAL | BREAK | (code << 6);
2441 void Assembler::stop(
const char* msg,
uint32_t code) {
2442 DCHECK_GT(code, kMaxWatchpointCode);
2443 DCHECK_LE(code, kMaxStopCode);
2444 #if V8_HOST_ARCH_MIPS 2446 #else // V8_HOST_ARCH_MIPS 2452 void Assembler::tge(Register rs, Register rt, uint16_t code) {
2453 DCHECK(is_uint10(code));
2454 Instr instr = SPECIAL | TGE | rs.code() << kRsShift
2455 | rt.code() << kRtShift | code << 6;
2460 void Assembler::tgeu(Register rs, Register rt, uint16_t code) {
2461 DCHECK(is_uint10(code));
2462 Instr instr = SPECIAL | TGEU | rs.code() << kRsShift
2463 | rt.code() << kRtShift | code << 6;
2468 void Assembler::tlt(Register rs, Register rt, uint16_t code) {
2469 DCHECK(is_uint10(code));
2471 SPECIAL | TLT | rs.code() << kRsShift | rt.code() << kRtShift | code << 6;
2476 void Assembler::tltu(Register rs, Register rt, uint16_t code) {
2477 DCHECK(is_uint10(code));
2479 SPECIAL | TLTU | rs.code() << kRsShift
2480 | rt.code() << kRtShift | code << 6;
2485 void Assembler::teq(Register rs, Register rt, uint16_t code) {
2486 DCHECK(is_uint10(code));
2488 SPECIAL | TEQ | rs.code() << kRsShift | rt.code() << kRtShift | code << 6;
2493 void Assembler::tne(Register rs, Register rt, uint16_t code) {
2494 DCHECK(is_uint10(code));
2496 SPECIAL | TNE | rs.code() << kRsShift | rt.code() << kRtShift | code << 6;
2500 void Assembler::sync() {
2501 Instr sync_instr = SPECIAL | SYNC;
2507 void Assembler::mfhi(Register rd) {
2508 GenInstrRegister(SPECIAL, zero_reg, zero_reg, rd, 0, MFHI);
2512 void Assembler::mflo(Register rd) {
2513 GenInstrRegister(SPECIAL, zero_reg, zero_reg, rd, 0, MFLO);
2518 void Assembler::slt(Register rd, Register rs, Register rt) {
2519 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SLT);
2523 void Assembler::sltu(Register rd, Register rs, Register rt) {
2524 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SLTU);
2528 void Assembler::slti(Register rt, Register rs, int32_t j) {
2529 GenInstrImmediate(SLTI, rs, rt, j);
2533 void Assembler::sltiu(Register rt, Register rs, int32_t j) {
2534 GenInstrImmediate(SLTIU, rs, rt, j);
2539 void Assembler::movz(Register rd, Register rs, Register rt) {
2540 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVZ);
2544 void Assembler::movn(Register rd, Register rs, Register rt) {
2545 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVN);
2549 void Assembler::movt(Register rd, Register rs, uint16_t cc) {
2550 Register rt = Register::from_code((cc & 0x0007) << 2 | 1);
2551 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVCI);
2555 void Assembler::movf(Register rd, Register rs, uint16_t cc) {
2556 Register rt = Register::from_code((cc & 0x0007) << 2 | 0);
2557 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVCI);
2561 void Assembler::seleqz(Register rd, Register rs, Register rt) {
2562 DCHECK(IsMipsArchVariant(kMips32r6));
2563 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SELEQZ_S);
2568 void Assembler::clz(Register rd, Register rs) {
2569 if (!IsMipsArchVariant(kMips32r6)) {
2571 GenInstrRegister(SPECIAL2, rs, rd, rd, 0, CLZ);
2573 GenInstrRegister(SPECIAL, rs, zero_reg, rd, 1, CLZ_R6);
2578 void Assembler::ins_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2581 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2582 GenInstrRegister(SPECIAL3, rs, rt, pos + size - 1, pos, INS);
2586 void Assembler::ext_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2589 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2590 GenInstrRegister(SPECIAL3, rs, rt, size - 1, pos, EXT);
2594 void Assembler::bitswap(Register rd, Register rt) {
2595 DCHECK(IsMipsArchVariant(kMips32r6));
2596 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, 0, BSHFL);
2600 void Assembler::pref(int32_t hint,
const MemOperand& rs) {
2601 DCHECK(!IsMipsArchVariant(kLoongson));
2602 DCHECK(is_uint5(hint) && is_uint16(rs.offset_));
2603 Instr instr = PREF | (rs.rm().code() << kRsShift) | (hint << kRtShift)
2609 void Assembler::align(Register rd, Register rs, Register rt, uint8_t bp) {
2610 DCHECK(IsMipsArchVariant(kMips32r6));
2611 DCHECK(is_uint3(bp));
2612 uint16_t sa = (ALIGN << kBp2Bits) | bp;
2613 GenInstrRegister(SPECIAL3, rs, rt, rd, sa, BSHFL);
2617 void Assembler::wsbh(Register rd, Register rt) {
2618 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2619 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, WSBH, BSHFL);
2622 void Assembler::seh(Register rd, Register rt) {
2623 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2624 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, SEH, BSHFL);
2627 void Assembler::seb(Register rd, Register rt) {
2628 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2629 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, SEB, BSHFL);
2635 void Assembler::lwc1(FPURegister fd,
const MemOperand& src) {
2636 MemOperand tmp = src;
2637 AdjustBaseAndOffset(tmp);
2638 GenInstrImmediate(LWC1, tmp.rm(), fd, tmp.offset());
2642 void Assembler::swc1(FPURegister fd,
const MemOperand& src) {
2643 MemOperand tmp = src;
2644 AdjustBaseAndOffset(tmp);
2645 GenInstrImmediate(SWC1, tmp.rm(), fd, tmp.offset());
2649 void Assembler::mtc1(Register rt, FPURegister fs) {
2650 GenInstrRegister(COP1, MTC1, rt, fs, f0);
2654 void Assembler::mthc1(Register rt, FPURegister fs) {
2655 GenInstrRegister(COP1, MTHC1, rt, fs, f0);
2659 void Assembler::mfc1(Register rt, FPURegister fs) {
2660 GenInstrRegister(COP1, MFC1, rt, fs, f0);
2664 void Assembler::mfhc1(Register rt, FPURegister fs) {
2665 GenInstrRegister(COP1, MFHC1, rt, fs, f0);
2669 void Assembler::ctc1(Register rt, FPUControlRegister fs) {
2670 GenInstrRegister(COP1, CTC1, rt, fs);
2674 void Assembler::cfc1(Register rt, FPUControlRegister fs) {
2675 GenInstrRegister(COP1, CFC1, rt, fs);
2679 void Assembler::movn_s(FPURegister fd, FPURegister fs, Register rt) {
2680 DCHECK(!IsMipsArchVariant(kMips32r6));
2681 GenInstrRegister(COP1, S, rt, fs, fd, MOVN_C);
2685 void Assembler::movn_d(FPURegister fd, FPURegister fs, Register rt) {
2686 DCHECK(!IsMipsArchVariant(kMips32r6));
2687 GenInstrRegister(COP1, D, rt, fs, fd, MOVN_C);
2691 void Assembler::sel(SecondaryField fmt, FPURegister fd, FPURegister fs,
2693 DCHECK(IsMipsArchVariant(kMips32r6));
2694 DCHECK((fmt == D) || (fmt == S));
2696 GenInstrRegister(COP1, fmt, ft, fs, fd,
SEL);
2700 void Assembler::sel_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2705 void Assembler::sel_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2710 void Assembler::seleqz(SecondaryField fmt, FPURegister fd, FPURegister fs,
2712 DCHECK(IsMipsArchVariant(kMips32r6));
2713 DCHECK((fmt == D) || (fmt == S));
2714 GenInstrRegister(COP1, fmt, ft, fs, fd, SELEQZ_C);
2718 void Assembler::selnez(Register rd, Register rs, Register rt) {
2719 DCHECK(IsMipsArchVariant(kMips32r6));
2720 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SELNEZ_S);
2724 void Assembler::selnez(SecondaryField fmt, FPURegister fd, FPURegister fs,
2726 DCHECK(IsMipsArchVariant(kMips32r6));
2727 DCHECK((fmt == D) || (fmt == S));
2728 GenInstrRegister(COP1, fmt, ft, fs, fd, SELNEZ_C);
2732 void Assembler::seleqz_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2733 seleqz(D, fd, fs, ft);
2737 void Assembler::seleqz_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2738 seleqz(S, fd, fs, ft);
2742 void Assembler::selnez_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2743 selnez(D, fd, fs, ft);
2747 void Assembler::selnez_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2748 selnez(S, fd, fs, ft);
2752 void Assembler::movz_s(FPURegister fd, FPURegister fs, Register rt) {
2753 DCHECK(!IsMipsArchVariant(kMips32r6));
2754 GenInstrRegister(COP1, S, rt, fs, fd, MOVZ_C);
2758 void Assembler::movz_d(FPURegister fd, FPURegister fs, Register rt) {
2759 DCHECK(!IsMipsArchVariant(kMips32r6));
2760 GenInstrRegister(COP1, D, rt, fs, fd, MOVZ_C);
2764 void Assembler::movt_s(FPURegister fd, FPURegister fs, uint16_t cc) {
2765 DCHECK(!IsMipsArchVariant(kMips32r6));
2766 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 1);
2767 GenInstrRegister(COP1, S, ft, fs, fd, MOVF);
2771 void Assembler::movt_d(FPURegister fd, FPURegister fs, uint16_t cc) {
2772 DCHECK(!IsMipsArchVariant(kMips32r6));
2773 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 1);
2774 GenInstrRegister(COP1, D, ft, fs, fd, MOVF);
2778 void Assembler::movf_s(FPURegister fd, FPURegister fs, uint16_t cc) {
2779 DCHECK(!IsMipsArchVariant(kMips32r6));
2780 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 0);
2781 GenInstrRegister(COP1, S, ft, fs, fd, MOVF);
2785 void Assembler::movf_d(FPURegister fd, FPURegister fs, uint16_t cc) {
2786 DCHECK(!IsMipsArchVariant(kMips32r6));
2787 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 0);
2788 GenInstrRegister(COP1, D, ft, fs, fd, MOVF);
2794 void Assembler::add_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2795 GenInstrRegister(COP1, S, ft, fs, fd, ADD_S);
2799 void Assembler::add_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2800 GenInstrRegister(COP1, D, ft, fs, fd, ADD_D);
2804 void Assembler::sub_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2805 GenInstrRegister(COP1, S, ft, fs, fd, SUB_S);
2809 void Assembler::sub_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2810 GenInstrRegister(COP1, D, ft, fs, fd, SUB_D);
2814 void Assembler::mul_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2815 GenInstrRegister(COP1, S, ft, fs, fd, MUL_S);
2819 void Assembler::mul_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2820 GenInstrRegister(COP1, D, ft, fs, fd, MUL_D);
2823 void Assembler::madd_s(FPURegister fd, FPURegister fr, FPURegister fs,
2825 DCHECK(IsMipsArchVariant(kMips32r2));
2826 GenInstrRegister(COP1X, fr, ft, fs, fd, MADD_S);
2829 void Assembler::madd_d(FPURegister fd, FPURegister fr, FPURegister fs,
2831 DCHECK(IsMipsArchVariant(kMips32r2));
2832 GenInstrRegister(COP1X, fr, ft, fs, fd, MADD_D);
2835 void Assembler::msub_s(FPURegister fd, FPURegister fr, FPURegister fs,
2837 DCHECK(IsMipsArchVariant(kMips32r2));
2838 GenInstrRegister(COP1X, fr, ft, fs, fd, MSUB_S);
2841 void Assembler::msub_d(FPURegister fd, FPURegister fr, FPURegister fs,
2843 DCHECK(IsMipsArchVariant(kMips32r2));
2844 GenInstrRegister(COP1X, fr, ft, fs, fd, MSUB_D);
2847 void Assembler::maddf_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2848 DCHECK(IsMipsArchVariant(kMips32r6));
2849 GenInstrRegister(COP1, S, ft, fs, fd, MADDF_S);
2852 void Assembler::maddf_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2853 DCHECK(IsMipsArchVariant(kMips32r6));
2854 GenInstrRegister(COP1, D, ft, fs, fd, MADDF_D);
2857 void Assembler::msubf_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2858 DCHECK(IsMipsArchVariant(kMips32r6));
2859 GenInstrRegister(COP1, S, ft, fs, fd, MSUBF_S);
2862 void Assembler::msubf_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2863 DCHECK(IsMipsArchVariant(kMips32r6));
2864 GenInstrRegister(COP1, D, ft, fs, fd, MSUBF_D);
2867 void Assembler::div_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2868 GenInstrRegister(COP1, S, ft, fs, fd, DIV_S);
2872 void Assembler::div_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2873 GenInstrRegister(COP1, D, ft, fs, fd, DIV_D);
2877 void Assembler::abs_s(FPURegister fd, FPURegister fs) {
2878 GenInstrRegister(COP1, S, f0, fs, fd, ABS_S);
2882 void Assembler::abs_d(FPURegister fd, FPURegister fs) {
2883 GenInstrRegister(COP1, D, f0, fs, fd, ABS_D);
2887 void Assembler::mov_d(FPURegister fd, FPURegister fs) {
2888 GenInstrRegister(COP1, D, f0, fs, fd, MOV_D);
2892 void Assembler::mov_s(FPURegister fd, FPURegister fs) {
2893 GenInstrRegister(COP1, S, f0, fs, fd, MOV_S);
2897 void Assembler::neg_s(FPURegister fd, FPURegister fs) {
2898 GenInstrRegister(COP1, S, f0, fs, fd, NEG_S);
2902 void Assembler::neg_d(FPURegister fd, FPURegister fs) {
2903 GenInstrRegister(COP1, D, f0, fs, fd, NEG_D);
2907 void Assembler::sqrt_s(FPURegister fd, FPURegister fs) {
2908 GenInstrRegister(COP1, S, f0, fs, fd, SQRT_S);
2912 void Assembler::sqrt_d(FPURegister fd, FPURegister fs) {
2913 GenInstrRegister(COP1, D, f0, fs, fd, SQRT_D);
2917 void Assembler::rsqrt_s(FPURegister fd, FPURegister fs) {
2918 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2919 GenInstrRegister(COP1, S, f0, fs, fd, RSQRT_S);
2923 void Assembler::rsqrt_d(FPURegister fd, FPURegister fs) {
2924 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2925 GenInstrRegister(COP1, D, f0, fs, fd, RSQRT_D);
2929 void Assembler::recip_d(FPURegister fd, FPURegister fs) {
2930 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2931 GenInstrRegister(COP1, D, f0, fs, fd, RECIP_D);
2935 void Assembler::recip_s(FPURegister fd, FPURegister fs) {
2936 DCHECK(IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6));
2937 GenInstrRegister(COP1, S, f0, fs, fd, RECIP_S);
2943 void Assembler::cvt_w_s(FPURegister fd, FPURegister fs) {
2944 GenInstrRegister(COP1, S, f0, fs, fd, CVT_W_S);
2948 void Assembler::cvt_w_d(FPURegister fd, FPURegister fs) {
2949 GenInstrRegister(COP1, D, f0, fs, fd, CVT_W_D);
2953 void Assembler::trunc_w_s(FPURegister fd, FPURegister fs) {
2954 GenInstrRegister(COP1, S, f0, fs, fd, TRUNC_W_S);
2958 void Assembler::trunc_w_d(FPURegister fd, FPURegister fs) {
2959 GenInstrRegister(COP1, D, f0, fs, fd, TRUNC_W_D);
2963 void Assembler::round_w_s(FPURegister fd, FPURegister fs) {
2964 GenInstrRegister(COP1, S, f0, fs, fd, ROUND_W_S);
2968 void Assembler::round_w_d(FPURegister fd, FPURegister fs) {
2969 GenInstrRegister(COP1, D, f0, fs, fd, ROUND_W_D);
2973 void Assembler::floor_w_s(FPURegister fd, FPURegister fs) {
2974 GenInstrRegister(COP1, S, f0, fs, fd, FLOOR_W_S);
2978 void Assembler::floor_w_d(FPURegister fd, FPURegister fs) {
2979 GenInstrRegister(COP1, D, f0, fs, fd, FLOOR_W_D);
2983 void Assembler::ceil_w_s(FPURegister fd, FPURegister fs) {
2984 GenInstrRegister(COP1, S, f0, fs, fd, CEIL_W_S);
2988 void Assembler::ceil_w_d(FPURegister fd, FPURegister fs) {
2989 GenInstrRegister(COP1, D, f0, fs, fd, CEIL_W_D);
2993 void Assembler::rint_s(FPURegister fd, FPURegister fs) { rint(S, fd, fs); }
2996 void Assembler::rint(SecondaryField fmt, FPURegister fd, FPURegister fs) {
2997 DCHECK(IsMipsArchVariant(kMips32r6));
2998 DCHECK((fmt == D) || (fmt == S));
2999 GenInstrRegister(COP1, fmt, f0, fs, fd, RINT);
3003 void Assembler::rint_d(FPURegister fd, FPURegister fs) { rint(D, fd, fs); }
3006 void Assembler::cvt_l_s(FPURegister fd, FPURegister fs) {
3007 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3009 GenInstrRegister(COP1, S, f0, fs, fd, CVT_L_S);
3013 void Assembler::cvt_l_d(FPURegister fd, FPURegister fs) {
3014 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3016 GenInstrRegister(COP1, D, f0, fs, fd, CVT_L_D);
3020 void Assembler::trunc_l_s(FPURegister fd, FPURegister fs) {
3021 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3023 GenInstrRegister(COP1, S, f0, fs, fd, TRUNC_L_S);
3027 void Assembler::trunc_l_d(FPURegister fd, FPURegister fs) {
3028 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3030 GenInstrRegister(COP1, D, f0, fs, fd, TRUNC_L_D);
3034 void Assembler::round_l_s(FPURegister fd, FPURegister fs) {
3035 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3037 GenInstrRegister(COP1, S, f0, fs, fd, ROUND_L_S);
3041 void Assembler::round_l_d(FPURegister fd, FPURegister fs) {
3042 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3044 GenInstrRegister(COP1, D, f0, fs, fd, ROUND_L_D);
3048 void Assembler::floor_l_s(FPURegister fd, FPURegister fs) {
3049 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3051 GenInstrRegister(COP1, S, f0, fs, fd, FLOOR_L_S);
3055 void Assembler::floor_l_d(FPURegister fd, FPURegister fs) {
3056 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3058 GenInstrRegister(COP1, D, f0, fs, fd, FLOOR_L_D);
3062 void Assembler::ceil_l_s(FPURegister fd, FPURegister fs) {
3063 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3065 GenInstrRegister(COP1, S, f0, fs, fd, CEIL_L_S);
3069 void Assembler::ceil_l_d(FPURegister fd, FPURegister fs) {
3070 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3072 GenInstrRegister(COP1, D, f0, fs, fd, CEIL_L_D);
3076 void Assembler::class_s(FPURegister fd, FPURegister fs) {
3077 DCHECK(IsMipsArchVariant(kMips32r6));
3078 GenInstrRegister(COP1, S, f0, fs, fd, CLASS_S);
3082 void Assembler::class_d(FPURegister fd, FPURegister fs) {
3083 DCHECK(IsMipsArchVariant(kMips32r6));
3084 GenInstrRegister(COP1, D, f0, fs, fd, CLASS_D);
3088 void Assembler::min(SecondaryField fmt, FPURegister fd, FPURegister fs,
3090 DCHECK(IsMipsArchVariant(kMips32r6));
3091 DCHECK((fmt == D) || (fmt == S));
3092 GenInstrRegister(COP1, fmt, ft, fs, fd, MIN);
3096 void Assembler::mina(SecondaryField fmt, FPURegister fd, FPURegister fs,
3098 DCHECK(IsMipsArchVariant(kMips32r6));
3099 DCHECK((fmt == D) || (fmt == S));
3100 GenInstrRegister(COP1, fmt, ft, fs, fd, MINA);
3104 void Assembler::max(SecondaryField fmt, FPURegister fd, FPURegister fs,
3106 DCHECK(IsMipsArchVariant(kMips32r6));
3107 DCHECK((fmt == D) || (fmt == S));
3108 GenInstrRegister(COP1, fmt, ft, fs, fd, MAX);
3112 void Assembler::maxa(SecondaryField fmt, FPURegister fd, FPURegister fs,
3114 DCHECK(IsMipsArchVariant(kMips32r6));
3115 DCHECK((fmt == D) || (fmt == S));
3116 GenInstrRegister(COP1, fmt, ft, fs, fd, MAXA);
3120 void Assembler::min_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3125 void Assembler::min_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3130 void Assembler::max_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3135 void Assembler::max_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3140 void Assembler::mina_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3141 mina(S, fd, fs, ft);
3145 void Assembler::mina_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3146 mina(D, fd, fs, ft);
3150 void Assembler::maxa_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3151 maxa(S, fd, fs, ft);
3155 void Assembler::maxa_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3156 maxa(D, fd, fs, ft);
3160 void Assembler::cvt_s_w(FPURegister fd, FPURegister fs) {
3161 GenInstrRegister(COP1, W, f0, fs, fd, CVT_S_W);
3165 void Assembler::cvt_s_l(FPURegister fd, FPURegister fs) {
3166 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3168 GenInstrRegister(COP1, L, f0, fs, fd, CVT_S_L);
3172 void Assembler::cvt_s_d(FPURegister fd, FPURegister fs) {
3173 GenInstrRegister(COP1, D, f0, fs, fd, CVT_S_D);
3177 void Assembler::cvt_d_w(FPURegister fd, FPURegister fs) {
3178 GenInstrRegister(COP1, W, f0, fs, fd, CVT_D_W);
3182 void Assembler::cvt_d_l(FPURegister fd, FPURegister fs) {
3183 DCHECK((IsMipsArchVariant(kMips32r2) || IsMipsArchVariant(kMips32r6)) &&
3185 GenInstrRegister(COP1, L, f0, fs, fd, CVT_D_L);
3189 void Assembler::cvt_d_s(FPURegister fd, FPURegister fs) {
3190 GenInstrRegister(COP1, S, f0, fs, fd, CVT_D_S);
3195 void Assembler::cmp(FPUCondition cond, SecondaryField fmt,
3196 FPURegister fd, FPURegister fs, FPURegister ft) {
3197 DCHECK(IsMipsArchVariant(kMips32r6));
3198 DCHECK_EQ(fmt & ~(31 << kRsShift), 0);
3199 Instr instr = COP1 | fmt | ft.code() << kFtShift |
3200 fs.code() << kFsShift | fd.code() << kFdShift | (0 << 5) | cond;
3205 void Assembler::cmp_s(FPUCondition cond, FPURegister fd, FPURegister fs,
3207 cmp(cond, W, fd, fs, ft);
3210 void Assembler::cmp_d(FPUCondition cond, FPURegister fd, FPURegister fs,
3212 cmp(cond, L, fd, fs, ft);
3216 void Assembler::bc1eqz(int16_t offset, FPURegister ft) {
3217 DCHECK(IsMipsArchVariant(kMips32r6));
3218 BlockTrampolinePoolScope block_trampoline_pool(
this);
3219 Instr instr = COP1 | BC1EQZ | ft.code() << kFtShift | (offset & kImm16Mask);
3221 BlockTrampolinePoolFor(1);
3225 void Assembler::bc1nez(int16_t offset, FPURegister ft) {
3226 DCHECK(IsMipsArchVariant(kMips32r6));
3227 BlockTrampolinePoolScope block_trampoline_pool(
this);
3228 Instr instr = COP1 | BC1NEZ | ft.code() << kFtShift | (offset & kImm16Mask);
3230 BlockTrampolinePoolFor(1);
3235 void Assembler::c(FPUCondition cond, SecondaryField fmt,
3236 FPURegister fs, FPURegister ft, uint16_t cc) {
3237 DCHECK(is_uint3(cc));
3238 DCHECK(fmt == S || fmt == D);
3239 DCHECK_EQ(fmt & ~(31 << kRsShift), 0);
3240 Instr instr = COP1 | fmt | ft.code() << 16 | fs.code() << kFsShift
3241 | cc << 8 | 3 << 4 | cond;
3246 void Assembler::c_s(FPUCondition cond, FPURegister fs, FPURegister ft,
3248 c(cond, S, fs, ft, cc);
3252 void Assembler::c_d(FPUCondition cond, FPURegister fs, FPURegister ft,
3254 c(cond, D, fs, ft, cc);
3258 void Assembler::fcmp(FPURegister src1,
const double src2,
3259 FPUCondition cond) {
3260 DCHECK_EQ(src2, 0.0);
3261 mtc1(zero_reg, f14);
3263 c(cond, D, src1, f14, 0);
3267 void Assembler::bc1f(int16_t offset, uint16_t cc) {
3268 BlockTrampolinePoolScope block_trampoline_pool(
this);
3269 DCHECK(is_uint3(cc));
3270 Instr instr = COP1 | BC1 | cc << 18 | 0 << 16 | (offset & kImm16Mask);
3272 BlockTrampolinePoolFor(1);
3276 void Assembler::bc1t(int16_t offset, uint16_t cc) {
3277 BlockTrampolinePoolScope block_trampoline_pool(
this);
3278 DCHECK(is_uint3(cc));
3279 Instr instr = COP1 | BC1 | cc << 18 | 1 << 16 | (offset & kImm16Mask);
3281 BlockTrampolinePoolFor(1);
3285 #define MSA_BRANCH_LIST(V) \ 3297 #define MSA_BRANCH(name, opcode) \ 3298 void Assembler::name(MSARegister wt, int16_t offset) { \ 3299 GenInstrMsaBranch(opcode, wt, offset); \ 3302 MSA_BRANCH_LIST(MSA_BRANCH)
3304 #undef MSA_BRANCH_LIST 3306 #define MSA_LD_ST_LIST(V) \ 3316 #define MSA_LD_ST(name, opcode) \ 3317 void Assembler::name(MSARegister wd, const MemOperand& rs) { \ 3318 MemOperand source = rs; \ 3319 AdjustBaseAndOffset(source); \ 3320 if (is_int10(source.offset())) { \ 3321 GenInstrMsaMI10(opcode, source.offset(), source.rm(), wd); \ 3323 UseScratchRegisterScope temps(this); \ 3324 Register scratch = temps.Acquire(); \ 3325 DCHECK(rs.rm() != scratch); \ 3326 addiu(scratch, source.rm(), source.offset()); \ 3327 GenInstrMsaMI10(opcode, 0, scratch, wd); \ 3331 MSA_LD_ST_LIST(MSA_LD_ST)
3333 #undef MSA_BRANCH_LIST 3335 #define MSA_I10_LIST(V) \ 3341 #define MSA_I10(name, format) \ 3342 void Assembler::name(MSARegister wd, int32_t imm10) { \ 3343 GenInstrMsaI10(LDI, format, imm10, wd); \ 3345 MSA_I10_LIST(MSA_I10)
3349 #define MSA_I5_LIST(V) \ 3362 #define MSA_I5_FORMAT(name, opcode, format) \ 3363 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 3365 GenInstrMsaI5(opcode, I5_DF_##format, imm5, ws, wd); \ 3368 #define MSA_I5(name, opcode) \ 3369 MSA_I5_FORMAT(name, opcode, b) \ 3370 MSA_I5_FORMAT(name, opcode, h) \ 3371 MSA_I5_FORMAT(name, opcode, w) \ 3372 MSA_I5_FORMAT(name, opcode, d) 3376 #undef MSA_I5_FORMAT 3379 #define MSA_I8_LIST(V) \ 3384 V(bmnzi_b, BMNZI_B) \ 3386 V(bseli_b, BSELI_B) \ 3391 #define MSA_I8(name, opcode) \ 3392 void Assembler::name(MSARegister wd, MSARegister ws, uint32_t imm8) { \ 3393 GenInstrMsaI8(opcode, imm8, ws, wd); \ 3400 #define MSA_VEC_LIST(V) \ 3409 #define MSA_VEC(name, opcode) \ 3410 void Assembler::name(MSARegister wd, MSARegister ws, MSARegister wt) { \ 3411 GenInstrMsaVec(opcode, wt, ws, wd); \ 3414 MSA_VEC_LIST(MSA_VEC)
3418 #define MSA_2R_LIST(V) \ 3423 #define MSA_2R_FORMAT(name, opcode, format) \ 3424 void Assembler::name##_##format(MSARegister wd, MSARegister ws) { \ 3425 GenInstrMsa2R(opcode, MSA_2R_DF_##format, ws, wd); \ 3428 #define MSA_2R(name, opcode) \ 3429 MSA_2R_FORMAT(name, opcode, b) \ 3430 MSA_2R_FORMAT(name, opcode, h) \ 3431 MSA_2R_FORMAT(name, opcode, w) \ 3432 MSA_2R_FORMAT(name, opcode, d) 3436 #undef MSA_2R_FORMAT 3439 #define MSA_FILL(format) \ 3440 void Assembler::fill_##format(MSARegister wd, Register rs) { \ 3441 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD)); \ 3442 DCHECK(rs.is_valid() && wd.is_valid()); \ 3443 Instr instr = MSA | MSA_2R_FORMAT | FILL | MSA_2R_DF_##format | \ 3444 (rs.code() << kWsShift) | (wd.code() << kWdShift) | \ 3445 MSA_VEC_2R_2RF_MINOR; \ 3454 #define MSA_2RF_LIST(V) \ 3456 V(ftrunc_s, FTRUNC_S) \ 3457 V(ftrunc_u, FTRUNC_U) \ 3467 V(ftint_s, FTINT_S) \ 3468 V(ftint_u, FTINT_U) \ 3469 V(ffint_s, FFINT_S) \ 3472 #define MSA_2RF_FORMAT(name, opcode, format) \ 3473 void Assembler::name##_##format(MSARegister wd, MSARegister ws) { \ 3474 GenInstrMsa2RF(opcode, MSA_2RF_DF_##format, ws, wd); \ 3477 #define MSA_2RF(name, opcode) \ 3478 MSA_2RF_FORMAT(name, opcode, w) \ 3479 MSA_2RF_FORMAT(name, opcode, d) 3481 MSA_2RF_LIST(MSA_2RF)
3483 #undef MSA_2RF_FORMAT 3486 #define MSA_3R_LIST(V) \ 3518 V(subsus_u, SUBSUS_U) \ 3519 V(subsuu_s, SUBSUU_S) \ 3525 V(div_s, DIV_S_MSA) \ 3531 V(dpadd_s, DPADD_S) \ 3532 V(dpadd_u, DPADD_U) \ 3533 V(dpsub_s, DPSUB_S) \ 3534 V(dpsub_u, DPSUB_U) \ 3549 #define MSA_3R_FORMAT(name, opcode, format) \ 3550 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 3552 GenInstrMsa3R<MSARegister>(opcode, MSA_3R_DF_##format, wt, ws, wd); \ 3555 #define MSA_3R_FORMAT_SLD_SPLAT(name, opcode, format) \ 3556 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 3558 GenInstrMsa3R<Register>(opcode, MSA_3R_DF_##format, rt, ws, wd); \ 3561 #define MSA_3R(name, opcode) \ 3562 MSA_3R_FORMAT(name, opcode, b) \ 3563 MSA_3R_FORMAT(name, opcode, h) \ 3564 MSA_3R_FORMAT(name, opcode, w) \ 3565 MSA_3R_FORMAT(name, opcode, d) 3567 #define MSA_3R_SLD_SPLAT(name, opcode) \ 3568 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, b) \ 3569 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, h) \ 3570 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, w) \ 3571 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, d) 3574 MSA_3R_SLD_SPLAT(sld, SLD)
3575 MSA_3R_SLD_SPLAT(splat, SPLAT)
3578 #undef MSA_3R_FORMAT 3579 #undef MSA_3R_FORMAT_SLD_SPLAT 3580 #undef MSA_3R_SLD_SPLAT 3583 #define MSA_3RF_LIST1(V) \ 3618 #define MSA_3RF_LIST2(V) \ 3625 V(maddr_q, MADDR_Q) \ 3628 #define MSA_3RF_FORMAT(name, opcode, df, df_c) \ 3629 void Assembler::name##_##df(MSARegister wd, MSARegister ws, \ 3631 GenInstrMsa3RF(opcode, df_c, wt, ws, wd); \ 3634 #define MSA_3RF_1(name, opcode) \ 3635 MSA_3RF_FORMAT(name, opcode, w, 0) \ 3636 MSA_3RF_FORMAT(name, opcode, d, 1) 3638 #define MSA_3RF_2(name, opcode) \ 3639 MSA_3RF_FORMAT(name, opcode, h, 0) \ 3640 MSA_3RF_FORMAT(name, opcode, w, 1) 3642 MSA_3RF_LIST1(MSA_3RF_1)
3643 MSA_3RF_LIST2(MSA_3RF_2)
3646 #undef MSA_3RF_FORMAT 3647 #undef MSA_3RF_LIST1 3648 #undef MSA_3RF_LIST2 3650 void Assembler::sldi_b(MSARegister wd, MSARegister ws,
uint32_t n) {
3651 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_B, n, ws, wd);
3654 void Assembler::sldi_h(MSARegister wd, MSARegister ws,
uint32_t n) {
3655 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_H, n, ws, wd);
3658 void Assembler::sldi_w(MSARegister wd, MSARegister ws,
uint32_t n) {
3659 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_W, n, ws, wd);
3662 void Assembler::sldi_d(MSARegister wd, MSARegister ws,
uint32_t n) {
3663 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_D, n, ws, wd);
3666 void Assembler::splati_b(MSARegister wd, MSARegister ws,
uint32_t n) {
3667 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_B, n, ws, wd);
3670 void Assembler::splati_h(MSARegister wd, MSARegister ws,
uint32_t n) {
3671 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_H, n, ws, wd);
3674 void Assembler::splati_w(MSARegister wd, MSARegister ws,
uint32_t n) {
3675 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_W, n, ws, wd);
3678 void Assembler::splati_d(MSARegister wd, MSARegister ws,
uint32_t n) {
3679 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_D, n, ws, wd);
3682 void Assembler::copy_s_b(Register rd, MSARegister ws,
uint32_t n) {
3683 GenInstrMsaElm<Register, MSARegister>(COPY_S, ELM_DF_B, n, ws, rd);
3686 void Assembler::copy_s_h(Register rd, MSARegister ws,
uint32_t n) {
3687 GenInstrMsaElm<Register, MSARegister>(COPY_S, ELM_DF_H, n, ws, rd);
3690 void Assembler::copy_s_w(Register rd, MSARegister ws,
uint32_t n) {
3691 GenInstrMsaElm<Register, MSARegister>(COPY_S, ELM_DF_W, n, ws, rd);
3694 void Assembler::copy_u_b(Register rd, MSARegister ws,
uint32_t n) {
3695 GenInstrMsaElm<Register, MSARegister>(COPY_U, ELM_DF_B, n, ws, rd);
3698 void Assembler::copy_u_h(Register rd, MSARegister ws,
uint32_t n) {
3699 GenInstrMsaElm<Register, MSARegister>(COPY_U, ELM_DF_H, n, ws, rd);
3702 void Assembler::copy_u_w(Register rd, MSARegister ws,
uint32_t n) {
3703 GenInstrMsaElm<Register, MSARegister>(COPY_U, ELM_DF_W, n, ws, rd);
3706 void Assembler::insert_b(MSARegister wd,
uint32_t n, Register rs) {
3707 GenInstrMsaElm<MSARegister, Register>(INSERT, ELM_DF_B, n, rs, wd);
3710 void Assembler::insert_h(MSARegister wd,
uint32_t n, Register rs) {
3711 GenInstrMsaElm<MSARegister, Register>(INSERT, ELM_DF_H, n, rs, wd);
3714 void Assembler::insert_w(MSARegister wd,
uint32_t n, Register rs) {
3715 GenInstrMsaElm<MSARegister, Register>(INSERT, ELM_DF_W, n, rs, wd);
3718 void Assembler::insve_b(MSARegister wd,
uint32_t n, MSARegister ws) {
3719 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_B, n, ws, wd);
3722 void Assembler::insve_h(MSARegister wd,
uint32_t n, MSARegister ws) {
3723 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_H, n, ws, wd);
3726 void Assembler::insve_w(MSARegister wd,
uint32_t n, MSARegister ws) {
3727 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_W, n, ws, wd);
3730 void Assembler::insve_d(MSARegister wd,
uint32_t n, MSARegister ws) {
3731 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_D, n, ws, wd);
3734 void Assembler::move_v(MSARegister wd, MSARegister ws) {
3735 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
3736 DCHECK(ws.is_valid() && wd.is_valid());
3737 Instr instr = MSA | MOVE_V | (ws.code() << kWsShift) |
3738 (wd.code() << kWdShift) | MSA_ELM_MINOR;
3742 void Assembler::ctcmsa(MSAControlRegister cd, Register rs) {
3743 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
3744 DCHECK(cd.is_valid() && rs.is_valid());
3745 Instr instr = MSA | CTCMSA | (rs.code() << kWsShift) |
3746 (cd.code() << kWdShift) | MSA_ELM_MINOR;
3750 void Assembler::cfcmsa(Register rd, MSAControlRegister cs) {
3751 DCHECK(IsMipsArchVariant(kMips32r6) && IsEnabled(MIPS_SIMD));
3752 DCHECK(rd.is_valid() && cs.is_valid());
3753 Instr instr = MSA | CFCMSA | (cs.code() << kWsShift) |
3754 (rd.code() << kWdShift) | MSA_ELM_MINOR;
3758 #define MSA_BIT_LIST(V) \ 3772 #define MSA_BIT_FORMAT(name, opcode, format) \ 3773 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 3775 GenInstrMsaBit(opcode, BIT_DF_##format, m, ws, wd); \ 3778 #define MSA_BIT(name, opcode) \ 3779 MSA_BIT_FORMAT(name, opcode, b) \ 3780 MSA_BIT_FORMAT(name, opcode, h) \ 3781 MSA_BIT_FORMAT(name, opcode, w) \ 3782 MSA_BIT_FORMAT(name, opcode, d) 3784 MSA_BIT_LIST(MSA_BIT)
3786 #undef MSA_BIT_FORMAT 3789 int Assembler::RelocateInternalReference(RelocInfo::Mode rmode, Address pc,
3790 intptr_t pc_delta) {
3791 Instr instr = instr_at(pc);
3793 if (RelocInfo::IsInternalReference(rmode)) {
3794 int32_t* p =
reinterpret_cast<int32_t*
>(pc);
3801 DCHECK(RelocInfo::IsInternalReferenceEncoded(rmode));
3803 Instr instr1 = instr_at(pc + 0 * kInstrSize);
3804 Instr instr2 = instr_at(pc + 1 * kInstrSize);
3805 DCHECK(IsOri(instr2) || IsJicOrJialc(instr2));
3807 if (IsJicOrJialc(instr2)) {
3808 imm = CreateTargetAddress(instr1, instr2);
3810 imm = GetLuiOriImmediate(instr1, instr2);
3813 if (imm == kEndOfJumpChain) {
3817 DCHECK_EQ(imm & 3, 0);
3818 instr1 &= ~kImm16Mask;
3819 instr2 &= ~kImm16Mask;
3821 if (IsJicOrJialc(instr2)) {
3822 uint32_t lui_offset_u, jic_offset_u;
3823 Assembler::UnpackTargetAddressUnsigned(imm, lui_offset_u, jic_offset_u);
3824 instr_at_put(pc + 0 * kInstrSize, instr1 | lui_offset_u);
3825 instr_at_put(pc + 1 * kInstrSize, instr2 | jic_offset_u);
3827 PatchLuiOriImmediate(pc, imm, instr1, 0 * kInstrSize, instr2,
3837 void Assembler::RelocateRelativeReference(RelocInfo::Mode rmode, Address pc,
3838 intptr_t pc_delta) {
3839 Instr instr = instr_at(pc);
3841 DCHECK(RelocInfo::IsRelativeCodeTarget(rmode));
3843 Instr instr1 = instr_at(pc + 0 * kInstrSize);
3844 Instr instr2 = instr_at(pc + 1 * kInstrSize);
3845 Instr instr3 = instr_at(pc + 2 * kInstrSize);
3848 if (IsNal(instr2)) {
3850 ori_offset = 2 * kInstrSize;
3852 ori_offset = 1 * kInstrSize;
3854 DCHECK(IsOri(instr2));
3855 imm = GetLuiOriImmediate(instr1, instr2);
3856 instr1 &= ~kImm16Mask;
3857 instr2 &= ~kImm16Mask;
3859 if (imm == kEndOfJumpChain) {
3863 DCHECK_EQ(imm & 3, 0);
3864 PatchLuiOriImmediate(pc, imm, instr1, 0 * kInstrSize, instr2, ori_offset);
3871 void Assembler::GrowBuffer() {
3872 if (!own_buffer_) FATAL(
"external code buffer is too small");
3876 if (buffer_size_ < 1 * MB) {
3877 desc.buffer_size = 2*buffer_size_;
3879 desc.buffer_size = buffer_size_ + 1*MB;
3884 if (desc.buffer_size > kMaximalBufferSize) {
3885 V8::FatalProcessOutOfMemory(
nullptr,
"Assembler::GrowBuffer");
3889 desc.buffer = NewArray<byte>(desc.buffer_size);
3892 desc.instr_size = pc_offset();
3893 desc.reloc_size = (buffer_ + buffer_size_) - reloc_info_writer.pos();
3896 int pc_delta = desc.buffer - buffer_;
3897 int rc_delta = (desc.buffer + desc.buffer_size) - (buffer_ + buffer_size_);
3898 MemMove(desc.buffer, buffer_, desc.instr_size);
3899 MemMove(reloc_info_writer.pos() + rc_delta, reloc_info_writer.pos(),
3903 DeleteArray(buffer_);
3904 buffer_ = desc.buffer;
3905 buffer_size_ = desc.buffer_size;
3907 reloc_info_writer.Reposition(reloc_info_writer.pos() + rc_delta,
3908 reloc_info_writer.last_pc() + pc_delta);
3911 for (RelocIterator it(desc); !it.done(); it.next()) {
3912 RelocInfo::Mode rmode = it.rinfo()->rmode();
3913 if (rmode == RelocInfo::INTERNAL_REFERENCE_ENCODED ||
3914 rmode == RelocInfo::INTERNAL_REFERENCE) {
3915 RelocateInternalReference(rmode, it.rinfo()->pc(), pc_delta);
3918 DCHECK(!overflow());
3922 void Assembler::db(uint8_t data) {
3923 CheckForEmitInForbiddenSlot();
3928 void Assembler::dd(
uint32_t data) {
3929 CheckForEmitInForbiddenSlot();
3934 void Assembler::dq(uint64_t data) {
3935 CheckForEmitInForbiddenSlot();
3940 void Assembler::dd(Label* label) {
3942 CheckForEmitInForbiddenSlot();
3943 if (label->is_bound()) {
3944 data =
reinterpret_cast<uint32_t>(buffer_ + label->pos());
3946 data = jump_address(label);
3947 unbound_labels_count_++;
3948 internal_reference_positions_.insert(label->pos());
3950 RecordRelocInfo(RelocInfo::INTERNAL_REFERENCE);
3955 void Assembler::RecordRelocInfo(RelocInfo::Mode rmode, intptr_t data) {
3956 if (!ShouldRecordRelocInfo(rmode))
return;
3958 RelocInfo rinfo(reinterpret_cast<Address>(pc_), rmode, data, Code());
3959 DCHECK_GE(buffer_space(), kMaxRelocSize);
3960 reloc_info_writer.Write(&rinfo);
3963 void Assembler::BlockTrampolinePoolFor(
int instructions) {
3964 CheckTrampolinePoolQuick(instructions);
3965 BlockTrampolinePoolBefore(pc_offset() + instructions * kInstrSize);
3969 void Assembler::CheckTrampolinePool() {
3975 if ((trampoline_pool_blocked_nesting_ > 0) ||
3976 (pc_offset() < no_trampoline_pool_before_)) {
3979 if (trampoline_pool_blocked_nesting_ > 0) {
3980 next_buffer_check_ = pc_offset() + kInstrSize;
3982 next_buffer_check_ = no_trampoline_pool_before_;
3987 DCHECK(!trampoline_emitted_);
3988 DCHECK_GE(unbound_labels_count_, 0);
3989 if (unbound_labels_count_ > 0) {
3991 { BlockTrampolinePoolScope block_trampoline_pool(
this);
3993 if (IsMipsArchVariant(kMips32r6)) {
4000 int pool_start = pc_offset();
4001 for (
int i = 0;
i < unbound_labels_count_;
i++) {
4003 if (IsMipsArchVariant(kMips32r6)) {
4007 GenPCRelativeJump(t8, t9, 0, RelocInfo::NONE,
4008 BranchDelaySlot::PROTECT);
4013 trampoline_ = Trampoline(pool_start, unbound_labels_count_);
4015 trampoline_emitted_ =
true;
4018 next_buffer_check_ = kMaxInt;
4023 next_buffer_check_ = pc_offset() +
4024 kMaxBranchOffset - kTrampolineSlotsSize * 16;
4030 Address Assembler::target_address_at(Address pc) {
4031 Instr instr1 = instr_at(pc);
4032 Instr instr2 = instr_at(pc + kInstrSize);
4033 Instr instr3 = instr_at(pc + 2 * kInstrSize);
4036 if (IsLui(instr1)) {
4037 if (IsOri(instr2)) {
4038 Address target_address;
4040 target_address = GetLuiOriImmediate(instr1, instr2);
4041 if (IsAddu(instr3, t9, ra, t9)) {
4042 target_address += pc + kRelativeJumpForBuiltinsOffset;
4044 return target_address;
4045 }
else if (IsJicOrJialc(instr2)) {
4047 return static_cast<Address
>(CreateTargetAddress(instr1, instr2));
4048 }
else if (IsNal(instr2)) {
4049 DCHECK(IsOri(instr3));
4050 Address target_address;
4051 target_address = GetLuiOriImmediate(instr1, instr3);
4052 return target_address + pc + kRelativeCallForBuiltinsOffset;
4065 void Assembler::QuietNaN(HeapObject*
object) {
4066 HeapNumber::cast(
object)->set_value(std::numeric_limits<double>::quiet_NaN());
4075 void Assembler::set_target_value_at(Address pc,
uint32_t target,
4076 ICacheFlushMode icache_flush_mode) {
4077 Instr instr1 = instr_at(pc);
4078 Instr instr2 = instr_at(pc + kInstrSize);
4082 DCHECK(IsLui(instr1) &&
4083 (IsOri(instr2) || IsJicOrJialc(instr2) || IsNal(instr2)));
4086 if (IsJicOrJialc(instr2)) {
4089 Assembler::UnpackTargetAddressUnsigned(target, lui_offset, jic_offset);
4091 instr1 &= ~kImm16Mask;
4092 instr2 &= ~kImm16Mask;
4094 instr1 |= lui_offset;
4095 instr2 |= jic_offset;
4097 instr_at_put(pc, instr1);
4098 instr_at_put(pc + kInstrSize, instr2);
4100 Instr instr3 = instr_at(pc + 2 * kInstrSize);
4102 if (IsNal(instr2)) {
4103 target -= pc + kRelativeCallForBuiltinsOffset;
4105 if (IsAddu(instr3, t9, ra, t9)) {
4106 target -= pc + kRelativeJumpForBuiltinsOffset;
4111 if (IsNal(instr2)) {
4112 instr1 &= ~kImm16Mask;
4113 instr3 &= ~kImm16Mask;
4114 PatchLuiOriImmediate(pc, target, instr1, 0 * kInstrSize, instr3,
4117 instr1 &= ~kImm16Mask;
4118 instr2 &= ~kImm16Mask;
4119 PatchLuiOriImmediate(pc, target, instr1, 0 * kInstrSize, instr2,
4124 if (icache_flush_mode != SKIP_ICACHE_FLUSH) {
4125 Assembler::FlushICache(pc, 2 *
sizeof(int32_t));
4129 void Assembler::GenPCRelativeJump(Register tf, Register ts, int32_t imm32,
4130 RelocInfo::Mode rmode,
4131 BranchDelaySlot bdslot) {
4134 or_(tf, ra, zero_reg);
4136 if (!RelocInfo::IsNone(rmode)) {
4137 RecordRelocInfo(rmode);
4139 lui(ts, (imm32 & kHiMask) >> kLuiShift);
4140 ori(ts, ts, (imm32 & kImm16Mask));
4142 if (bdslot == USE_DELAY_SLOT) {
4143 or_(ra, tf, zero_reg);
4146 if (bdslot == PROTECT) {
4147 or_(ra, tf, zero_reg);
4151 void Assembler::GenPCRelativeJumpAndLink(Register t, int32_t imm32,
4152 RelocInfo::Mode rmode,
4153 BranchDelaySlot bdslot) {
4154 if (!RelocInfo::IsNone(rmode)) {
4155 RecordRelocInfo(rmode);
4159 lui(t, (imm32 & kHiMask) >> kLuiShift);
4161 ori(t, t, (imm32 & kImm16Mask));
4164 if (bdslot == PROTECT) nop();
4167 UseScratchRegisterScope::UseScratchRegisterScope(Assembler* assembler)
4168 : available_(assembler->GetScratchRegisterList()),
4169 old_available_(*available_) {}
4171 UseScratchRegisterScope::~UseScratchRegisterScope() {
4172 *available_ = old_available_;
4175 Register UseScratchRegisterScope::Acquire() {
4176 DCHECK_NOT_NULL(available_);
4177 DCHECK_NE(*available_, 0);
4178 int index =
static_cast<int>(base::bits::CountTrailingZeros32(*available_));
4179 *available_ &= ~(1UL << index);
4181 return Register::from_code(index);
4184 bool UseScratchRegisterScope::hasAvailable()
const {
return *available_ != 0; }
4189 #endif // V8_TARGET_ARCH_MIPS