35 #include "src/mips64/assembler-mips64.h" 37 #if V8_TARGET_ARCH_MIPS64 39 #include "src/base/cpu.h" 40 #include "src/code-stubs.h" 41 #include "src/deoptimizer.h" 42 #include "src/mips64/assembler-mips64-inl.h" 43 #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_MIPS64R6) && defined(_MIPS_MSA) 82 supported_ |= 1u << MIPS_SIMD;
87 if (cpu.has_fpu()) supported_ |= 1u << FPU;
88 #if defined(_MIPS_ARCH_MIPS64R6) 89 #if defined(_MIPS_MSA) 90 supported_ |= 1u << MIPS_SIMD;
92 if (cpu.has_msa()) supported_ |= 1u << MIPS_SIMD;
99 void CpuFeatures::PrintTarget() { }
100 void CpuFeatures::PrintFeatures() { }
103 int ToNumber(Register reg) {
104 DCHECK(reg.is_valid());
105 const int kNumbers[] = {
139 return kNumbers[reg.code()];
143 Register ToRegister(
int num) {
144 DCHECK(num >= 0 && num < kNumRegisters);
145 const Register kRegisters[] = {
149 a0, a1, a2, a3, a4, a5, a6, a7,
151 s0, s1, s2, s3, s4, s5, s6, s7,
159 return kRegisters[num];
166 const int RelocInfo::kApplyMask =
167 RelocInfo::ModeMask(RelocInfo::INTERNAL_REFERENCE) |
168 RelocInfo::ModeMask(RelocInfo::INTERNAL_REFERENCE_ENCODED);
170 bool RelocInfo::IsCodedSpecially() {
178 bool RelocInfo::IsInConstantPool() {
182 int RelocInfo::GetDeoptimizationId(Isolate* isolate, DeoptimizeKind kind) {
183 DCHECK(IsRuntimeEntry(rmode_));
184 return Deoptimizer::GetDeoptimizationId(isolate, target_address(), kind);
187 uint32_t RelocInfo::wasm_call_tag()
const {
188 DCHECK(rmode_ == WASM_CALL || rmode_ == WASM_STUB_CALL);
190 Assembler::target_address_at(pc_, constant_pool_));
197 Operand::Operand(Handle<HeapObject> handle)
198 : rm_(no_reg), rmode_(RelocInfo::EMBEDDED_OBJECT) {
199 value_.immediate =
static_cast<intptr_t
>(handle.address());
202 Operand Operand::EmbeddedNumber(
double value) {
204 if (DoubleToSmiInteger(value, &smi))
return Operand(Smi::FromInt(smi));
205 Operand result(0, RelocInfo::EMBEDDED_OBJECT);
206 result.is_heap_object_request_ =
true;
207 result.value_.heap_object_request = HeapObjectRequest(value);
211 Operand Operand::EmbeddedCode(CodeStub* stub) {
212 Operand result(0, RelocInfo::CODE_TARGET);
213 result.is_heap_object_request_ =
true;
214 result.value_.heap_object_request = HeapObjectRequest(stub);
218 Operand Operand::EmbeddedStringConstant(
const StringConstantBase* str) {
219 Operand result(0, RelocInfo::EMBEDDED_OBJECT);
220 result.is_heap_object_request_ =
true;
221 result.value_.heap_object_request = HeapObjectRequest(str);
225 MemOperand::MemOperand(Register rm, int32_t offset) : Operand(rm) {
230 MemOperand::MemOperand(Register rm, int32_t unit, int32_t multiplier,
231 OffsetAddend offset_addend)
233 offset_ = unit * multiplier + offset_addend;
236 void Assembler::AllocateAndInstallRequestedHeapObjects(Isolate* isolate) {
237 DCHECK_IMPLIES(isolate ==
nullptr, heap_object_requests_.empty());
238 for (
auto& request : heap_object_requests_) {
239 Handle<HeapObject> object;
240 switch (request.kind()) {
241 case HeapObjectRequest::kHeapNumber:
243 isolate->factory()->NewHeapNumber(request.heap_number(), TENURED);
245 case HeapObjectRequest::kCodeStub:
246 request.code_stub()->set_isolate(isolate);
247 object = request.code_stub()->GetCode();
249 case HeapObjectRequest::kStringConstant:
250 const StringConstantBase* str = request.string();
252 object = str->AllocateStringConstant(isolate);
255 Address pc =
reinterpret_cast<Address
>(buffer_) + request.offset();
256 set_target_value_at(pc, reinterpret_cast<uint64_t>(
object.location()));
265 const Instr kPopInstruction = DADDIU | (sp.code() << kRsShift) |
266 (sp.code() << kRtShift) |
267 (kPointerSize & kImm16Mask);
269 const Instr kPushInstruction = DADDIU | (sp.code() << kRsShift) |
270 (sp.code() << kRtShift) |
271 (-kPointerSize & kImm16Mask);
273 const Instr kPushRegPattern =
274 SD | (sp.code() << kRsShift) | (0 & kImm16Mask);
276 const Instr kPopRegPattern =
277 LD | (sp.code() << kRsShift) | (0 & kImm16Mask);
279 const Instr kLwRegFpOffsetPattern =
280 LW | (fp.code() << kRsShift) | (0 & kImm16Mask);
282 const Instr kSwRegFpOffsetPattern =
283 SW | (fp.code() << kRsShift) | (0 & kImm16Mask);
285 const Instr kLwRegFpNegOffsetPattern =
286 LW | (fp.code() << kRsShift) | (kNegOffset & kImm16Mask);
288 const Instr kSwRegFpNegOffsetPattern =
289 SW | (fp.code() << kRsShift) | (kNegOffset & kImm16Mask);
291 const Instr kRtMask = kRtFieldMask;
292 const Instr kLwSwInstrTypeMask = 0xFFE00000;
293 const Instr kLwSwInstrArgumentMask = ~kLwSwInstrTypeMask;
294 const Instr kLwSwOffsetMask = kImm16Mask;
296 Assembler::Assembler(
const AssemblerOptions& options,
void* buffer,
298 : AssemblerBase(options, buffer, buffer_size),
299 scratch_register_list_(at.bit()) {
300 reloc_info_writer.Reposition(buffer_ + buffer_size_, pc_);
302 last_trampoline_pool_end_ = 0;
303 no_trampoline_pool_before_ = 0;
304 trampoline_pool_blocked_nesting_ = 0;
307 next_buffer_check_ = FLAG_force_long_branches
308 ? kMaxInt : kMaxBranchOffset - kTrampolineSlotsSize * 16;
309 internal_trampoline_exception_ =
false;
312 trampoline_emitted_ = FLAG_force_long_branches;
313 unbound_labels_count_ = 0;
314 block_buffer_growth_ =
false;
317 void Assembler::GetCode(Isolate* isolate, CodeDesc* desc) {
318 EmitForbiddenSlotInstruction();
319 DCHECK(pc_ <= reloc_info_writer.pos());
321 AllocateAndInstallRequestedHeapObjects(isolate);
324 desc->buffer = buffer_;
325 desc->buffer_size = buffer_size_;
326 desc->instr_size = pc_offset();
328 static_cast<int>((buffer_ + buffer_size_) - reloc_info_writer.pos());
330 desc->constant_pool_size = 0;
331 desc->unwinding_info_size = 0;
332 desc->unwinding_info =
nullptr;
336 void Assembler::Align(
int m) {
337 DCHECK(m >= 4 && base::bits::IsPowerOfTwo(m));
338 EmitForbiddenSlotInstruction();
339 while ((pc_offset() & (m - 1)) != 0) {
345 void Assembler::CodeTargetAlign() {
352 Register Assembler::GetRtReg(Instr instr) {
353 return Register::from_code((instr & kRtFieldMask) >> kRtShift);
357 Register Assembler::GetRsReg(Instr instr) {
358 return Register::from_code((instr & kRsFieldMask) >> kRsShift);
362 Register Assembler::GetRdReg(Instr instr) {
363 return Register::from_code((instr & kRdFieldMask) >> kRdShift);
367 uint32_t Assembler::GetRt(Instr instr) {
368 return (instr & kRtFieldMask) >> kRtShift;
372 uint32_t Assembler::GetRtField(Instr instr) {
373 return instr & kRtFieldMask;
377 uint32_t Assembler::GetRs(Instr instr) {
378 return (instr & kRsFieldMask) >> kRsShift;
382 uint32_t Assembler::GetRsField(Instr instr) {
383 return instr & kRsFieldMask;
387 uint32_t Assembler::GetRd(Instr instr) {
388 return (instr & kRdFieldMask) >> kRdShift;
392 uint32_t Assembler::GetRdField(Instr instr) {
393 return instr & kRdFieldMask;
397 uint32_t Assembler::GetSa(Instr instr) {
398 return (instr & kSaFieldMask) >> kSaShift;
402 uint32_t Assembler::GetSaField(Instr instr) {
403 return instr & kSaFieldMask;
407 uint32_t Assembler::GetOpcodeField(Instr instr) {
408 return instr & kOpcodeMask;
412 uint32_t Assembler::GetFunction(Instr instr) {
413 return (instr & kFunctionFieldMask) >> kFunctionShift;
417 uint32_t Assembler::GetFunctionField(Instr instr) {
418 return instr & kFunctionFieldMask;
422 uint32_t Assembler::GetImmediate16(Instr instr) {
423 return instr & kImm16Mask;
427 uint32_t Assembler::GetLabelConst(Instr instr) {
428 return instr & ~kImm16Mask;
432 bool Assembler::IsPop(Instr instr) {
433 return (instr & ~kRtMask) == kPopRegPattern;
437 bool Assembler::IsPush(Instr instr) {
438 return (instr & ~kRtMask) == kPushRegPattern;
442 bool Assembler::IsSwRegFpOffset(Instr instr) {
443 return ((instr & kLwSwInstrTypeMask) == kSwRegFpOffsetPattern);
447 bool Assembler::IsLwRegFpOffset(Instr instr) {
448 return ((instr & kLwSwInstrTypeMask) == kLwRegFpOffsetPattern);
452 bool Assembler::IsSwRegFpNegOffset(Instr instr) {
453 return ((instr & (kLwSwInstrTypeMask | kNegOffset)) ==
454 kSwRegFpNegOffsetPattern);
458 bool Assembler::IsLwRegFpNegOffset(Instr instr) {
459 return ((instr & (kLwSwInstrTypeMask | kNegOffset)) ==
460 kLwRegFpNegOffsetPattern);
479 const int kEndOfChain = -4;
481 const int kEndOfJumpChain = 0;
483 bool Assembler::IsMsaBranch(Instr instr) {
484 uint32_t opcode = GetOpcodeField(instr);
485 uint32_t rs_field = GetRsField(instr);
486 if (opcode == COP1) {
507 bool Assembler::IsBranch(Instr instr) {
508 uint32_t opcode = GetOpcodeField(instr);
509 uint32_t rt_field = GetRtField(instr);
510 uint32_t rs_field = GetRsField(instr);
513 opcode == BEQ || opcode == BNE || opcode == BLEZ || opcode == BGTZ ||
514 opcode == BEQL || opcode == BNEL || opcode == BLEZL || opcode == BGTZL ||
515 (opcode == REGIMM && (rt_field == BLTZ || rt_field == BGEZ ||
516 rt_field == BLTZAL || rt_field == BGEZAL)) ||
517 (opcode == COP1 && rs_field == BC1) ||
518 (opcode == COP1 && rs_field == BC1EQZ) ||
519 (opcode == COP1 && rs_field == BC1NEZ) || IsMsaBranch(instr);
520 if (!isBranch && kArchVariant == kMips64r6) {
523 isBranch |= opcode == POP10 || opcode == POP30 || opcode == BC ||
525 (opcode == POP66 && rs_field != 0) ||
526 (opcode == POP76 && rs_field != 0);
532 bool Assembler::IsBc(Instr instr) {
533 uint32_t opcode = GetOpcodeField(instr);
535 return opcode == BC || opcode == BALC;
538 bool Assembler::IsNal(Instr instr) {
539 uint32_t opcode = GetOpcodeField(instr);
540 uint32_t rt_field = GetRtField(instr);
541 uint32_t rs_field = GetRsField(instr);
542 return opcode == REGIMM && rt_field == BLTZAL && rs_field == 0;
545 bool Assembler::IsBzc(Instr instr) {
546 uint32_t opcode = GetOpcodeField(instr);
548 return (opcode == POP66 && GetRsField(instr) != 0) ||
549 (opcode == POP76 && GetRsField(instr) != 0);
553 bool Assembler::IsEmittedConstant(Instr instr) {
554 uint32_t label_constant = GetLabelConst(instr);
555 return label_constant == 0;
559 bool Assembler::IsBeq(Instr instr) {
560 return GetOpcodeField(instr) == BEQ;
564 bool Assembler::IsBne(Instr instr) {
565 return GetOpcodeField(instr) == BNE;
569 bool Assembler::IsBeqzc(Instr instr) {
570 uint32_t opcode = GetOpcodeField(instr);
571 return opcode == POP66 && GetRsField(instr) != 0;
575 bool Assembler::IsBnezc(Instr instr) {
576 uint32_t opcode = GetOpcodeField(instr);
577 return opcode == POP76 && GetRsField(instr) != 0;
581 bool Assembler::IsBeqc(Instr instr) {
582 uint32_t opcode = GetOpcodeField(instr);
585 return opcode == POP10 && rs != 0 && rs < rt;
589 bool Assembler::IsBnec(Instr instr) {
590 uint32_t opcode = GetOpcodeField(instr);
593 return opcode == POP30 && rs != 0 && rs < rt;
596 bool Assembler::IsMov(Instr instr, Register rd, Register rs) {
597 uint32_t opcode = GetOpcodeField(instr);
603 uint32_t function_field = GetFunctionField(instr);
605 bool res = opcode == SPECIAL && function_field == OR && rd_field == rd_reg &&
606 rs_field == rs_reg && rt_field == 0;
610 bool Assembler::IsJump(Instr instr) {
611 uint32_t opcode = GetOpcodeField(instr);
612 uint32_t rt_field = GetRtField(instr);
613 uint32_t rd_field = GetRdField(instr);
614 uint32_t function_field = GetFunctionField(instr);
616 return opcode == J || opcode == JAL ||
617 (opcode == SPECIAL && rt_field == 0 &&
618 ((function_field == JALR) || (rd_field == 0 && (function_field == JR))));
622 bool Assembler::IsJ(Instr instr) {
623 uint32_t opcode = GetOpcodeField(instr);
629 bool Assembler::IsJal(Instr instr) {
630 return GetOpcodeField(instr) == JAL;
634 bool Assembler::IsJr(Instr instr) {
635 return GetOpcodeField(instr) == SPECIAL && GetFunctionField(instr) == JR;
639 bool Assembler::IsJalr(Instr instr) {
640 return GetOpcodeField(instr) == SPECIAL && GetFunctionField(instr) == JALR;
644 bool Assembler::IsLui(Instr instr) {
645 uint32_t opcode = GetOpcodeField(instr);
647 return opcode == LUI;
651 bool Assembler::IsOri(Instr instr) {
652 uint32_t opcode = GetOpcodeField(instr);
654 return opcode == ORI;
658 bool Assembler::IsNop(Instr instr,
unsigned int type) {
661 uint32_t opcode = GetOpcodeField(instr);
662 uint32_t function = GetFunctionField(instr);
672 Register nop_rt_reg = (type == 0) ? zero_reg : at;
673 bool ret = (opcode == SPECIAL &&
function == SLL &&
674 rd ==
static_cast<uint32_t>(ToNumber(zero_reg)) &&
675 rt == static_cast<uint32_t>(ToNumber(nop_rt_reg)) &&
682 int32_t Assembler::GetBranchOffset(Instr instr) {
683 DCHECK(IsBranch(instr));
684 return (static_cast<int16_t>(instr & kImm16Mask)) << 2;
688 bool Assembler::IsLw(Instr instr) {
689 return (static_cast<uint32_t>(instr & kOpcodeMask) == LW);
693 int16_t Assembler::GetLwOffset(Instr instr) {
695 return ((instr & kImm16Mask));
699 Instr Assembler::SetLwOffset(Instr instr, int16_t offset) {
703 Instr temp_instr = LW | (instr & kRsFieldMask) | (instr & kRtFieldMask)
704 | (offset & kImm16Mask);
710 bool Assembler::IsSw(Instr instr) {
711 return (static_cast<uint32_t>(instr & kOpcodeMask) == SW);
715 Instr Assembler::SetSwOffset(Instr instr, int16_t offset) {
717 return ((instr & ~kImm16Mask) | (offset & kImm16Mask));
721 bool Assembler::IsAddImmediate(Instr instr) {
722 return ((instr & kOpcodeMask) == ADDIU || (instr & kOpcodeMask) == DADDIU);
726 Instr Assembler::SetAddImmediateOffset(Instr instr, int16_t offset) {
727 DCHECK(IsAddImmediate(instr));
728 return ((instr & ~kImm16Mask) | (offset & kImm16Mask));
732 bool Assembler::IsAndImmediate(Instr instr) {
733 return GetOpcodeField(instr) == ANDI;
737 static Assembler::OffsetSize OffsetSizeInBits(Instr instr) {
738 if (kArchVariant == kMips64r6) {
739 if (Assembler::IsBc(instr)) {
740 return Assembler::OffsetSize::kOffset26;
741 }
else if (Assembler::IsBzc(instr)) {
742 return Assembler::OffsetSize::kOffset21;
745 return Assembler::OffsetSize::kOffset16;
749 static inline int32_t AddBranchOffset(
int pos, Instr instr) {
750 int bits = OffsetSizeInBits(instr);
751 const int32_t mask = (1 << bits) - 1;
756 int32_t imm = ((instr & mask) << bits) >> (bits - 2);
758 if (imm == kEndOfChain) {
762 return pos + Assembler::kBranchPCOffset + imm;
767 int Assembler::target_at(
int pos,
bool is_internal) {
771 if (address == kEndOfJumpChain) {
775 DCHECK(instr_address - address < INT_MAX);
776 int delta =
static_cast<int>(instr_address - address);
781 Instr instr = instr_at(pos);
782 if ((instr & ~kImm16Mask) == 0) {
787 int32_t imm18 =((instr &
static_cast<int32_t
>(kImm16Mask)) << 16) >> 14;
788 return (imm18 + pos);
792 DCHECK(IsBranch(instr) || IsJ(instr) || IsJal(instr) || IsLui(instr) ||
793 IsMov(instr, t8, ra));
796 if (IsBranch(instr)) {
797 return AddBranchOffset(pos, instr);
798 }
else if (IsMov(instr, t8, ra)) {
800 Instr instr_lui = instr_at(pos + 2 * kInstrSize);
801 Instr instr_ori = instr_at(pos + 3 * kInstrSize);
802 DCHECK(IsLui(instr_lui));
803 DCHECK(IsOri(instr_ori));
804 imm32 = (instr_lui &
static_cast<int32_t
>(kImm16Mask)) << kLuiShift;
805 imm32 |= (instr_ori &
static_cast<int32_t
>(kImm16Mask));
806 if (imm32 == kEndOfJumpChain) {
810 return pos + Assembler::kLongBranchPCOffset + imm32;
811 }
else if (IsLui(instr)) {
812 if (IsNal(instr_at(pos + kInstrSize))) {
814 Instr instr_lui = instr_at(pos + 0 * kInstrSize);
815 Instr instr_ori = instr_at(pos + 2 * kInstrSize);
816 DCHECK(IsLui(instr_lui));
817 DCHECK(IsOri(instr_ori));
818 imm32 = (instr_lui &
static_cast<int32_t
>(kImm16Mask)) << kLuiShift;
819 imm32 |= (instr_ori &
static_cast<int32_t
>(kImm16Mask));
820 if (imm32 == kEndOfJumpChain) {
824 return pos + Assembler::kLongBranchPCOffset + imm32;
826 Instr instr_lui = instr_at(pos + 0 * kInstrSize);
827 Instr instr_ori = instr_at(pos + 1 * kInstrSize);
828 Instr instr_ori2 = instr_at(pos + 3 * kInstrSize);
829 DCHECK(IsOri(instr_ori));
830 DCHECK(IsOri(instr_ori2));
834 imm |=
static_cast<int64_t>(instr_ori & kImm16Mask) << 32;
835 imm |=
static_cast<int64_t>(instr_ori2 & kImm16Mask) << 16;
839 if (imm == kEndOfJumpChain) {
843 uint64_t instr_address =
reinterpret_cast<int64_t>(buffer_ + pos);
844 DCHECK(instr_address - imm < INT_MAX);
845 int delta =
static_cast<int>(instr_address - imm);
851 DCHECK(IsJ(instr) || IsJal(instr));
852 int32_t imm28 = (instr &
static_cast<int32_t
>(kImm26Mask)) << 2;
853 if (imm28 == kEndOfJumpChain) {
858 int32_t delta =
static_cast<int32_t
>((imm28 << 4) >> 4);
865 static inline Instr SetBranchOffset(int32_t pos, int32_t target_pos,
867 int32_t bits = OffsetSizeInBits(instr);
868 int32_t imm = target_pos - (pos + Assembler::kBranchPCOffset);
869 DCHECK_EQ(imm & 3, 0);
872 const int32_t mask = (1 << bits) - 1;
874 DCHECK(is_intn(imm, bits));
876 return instr | (imm & mask);
880 void Assembler::target_at_put(
int pos,
int target_pos,
bool is_internal) {
882 uint64_t imm =
reinterpret_cast<uint64_t
>(buffer_) + target_pos;
883 *
reinterpret_cast<uint64_t*
>(buffer_ + pos) = imm;
886 Instr instr = instr_at(pos);
887 if ((instr & ~kImm16Mask) == 0) {
888 DCHECK(target_pos == kEndOfChain || target_pos >= 0);
891 instr_at_put(pos, target_pos + (Code::kHeaderSize - kHeapObjectTag));
895 if (IsBranch(instr)) {
896 instr = SetBranchOffset(pos, target_pos, instr);
897 instr_at_put(pos, instr);
898 }
else if (IsLui(instr)) {
899 if (IsNal(instr_at(pos + kInstrSize))) {
900 Instr instr_lui = instr_at(pos + 0 * kInstrSize);
901 Instr instr_ori = instr_at(pos + 2 * kInstrSize);
902 DCHECK(IsLui(instr_lui));
903 DCHECK(IsOri(instr_ori));
904 int32_t imm = target_pos - (pos + Assembler::kLongBranchPCOffset);
905 DCHECK_EQ(imm & 3, 0);
906 if (is_int16(imm + Assembler::kLongBranchPCOffset -
907 Assembler::kBranchPCOffset)) {
910 Instr instr_b = REGIMM | BGEZAL;
911 instr_b = SetBranchOffset(pos, target_pos, instr_b);
914 Instr instr_a = DADDIU | ra.code() << kRsShift | ra.code() << kRtShift |
915 kOptimizedBranchAndLinkLongReturnOffset;
917 instr_at_put(pos, instr_b);
918 instr_at_put(pos + 1 * kInstrSize, instr_a);
920 instr_lui &= ~kImm16Mask;
921 instr_ori &= ~kImm16Mask;
923 instr_at_put(pos + 0 * kInstrSize,
924 instr_lui | ((imm >> kLuiShift) & kImm16Mask));
925 instr_at_put(pos + 2 * kInstrSize, instr_ori | (imm & kImm16Mask));
928 Instr instr_lui = instr_at(pos + 0 * kInstrSize);
929 Instr instr_ori = instr_at(pos + 1 * kInstrSize);
930 Instr instr_ori2 = instr_at(pos + 3 * kInstrSize);
931 DCHECK(IsOri(instr_ori));
932 DCHECK(IsOri(instr_ori2));
934 uint64_t imm =
reinterpret_cast<uint64_t
>(buffer_) + target_pos;
935 DCHECK_EQ(imm & 3, 0);
937 instr_lui &= ~kImm16Mask;
938 instr_ori &= ~kImm16Mask;
939 instr_ori2 &= ~kImm16Mask;
941 instr_at_put(pos + 0 * kInstrSize,
942 instr_lui | ((imm >> 32) & kImm16Mask));
943 instr_at_put(pos + 1 * kInstrSize,
944 instr_ori | ((imm >> 16) & kImm16Mask));
945 instr_at_put(pos + 3 * kInstrSize, instr_ori2 | (imm & kImm16Mask));
947 }
else if (IsMov(instr, t8, ra)) {
948 Instr instr_lui = instr_at(pos + 2 * kInstrSize);
949 Instr instr_ori = instr_at(pos + 3 * kInstrSize);
950 DCHECK(IsLui(instr_lui));
951 DCHECK(IsOri(instr_ori));
953 int32_t imm_short = target_pos - (pos + Assembler::kBranchPCOffset);
955 if (is_int16(imm_short)) {
959 instr_b = SetBranchOffset(pos, target_pos, instr_b);
961 Instr instr_j = instr_at(pos + 5 * kInstrSize);
962 Instr instr_branch_delay;
964 if (IsJump(instr_j)) {
965 instr_branch_delay = instr_at(pos + 6 * kInstrSize);
967 instr_branch_delay = instr_at(pos + 7 * kInstrSize);
969 instr_at_put(pos, instr_b);
970 instr_at_put(pos + 1 * kInstrSize, instr_branch_delay);
972 int32_t imm = target_pos - (pos + Assembler::kLongBranchPCOffset);
973 DCHECK_EQ(imm & 3, 0);
975 instr_lui &= ~kImm16Mask;
976 instr_ori &= ~kImm16Mask;
978 instr_at_put(pos + 2 * kInstrSize,
979 instr_lui | ((imm >> kLuiShift) & kImm16Mask));
980 instr_at_put(pos + 3 * kInstrSize, instr_ori | (imm & kImm16Mask));
982 }
else if (IsJ(instr) || IsJal(instr)) {
983 int32_t imm28 = target_pos - pos;
984 DCHECK_EQ(imm28 & 3, 0);
987 DCHECK(is_uint26(imm26));
990 int32_t mark = IsJ(instr) ? kJRawMark : kJalRawMark;
991 instr_at_put(pos, mark | (imm26 & kImm26Mask));
993 int32_t imm28 = target_pos - pos;
994 DCHECK_EQ(imm28 & 3, 0);
997 DCHECK(is_uint26(imm26));
1000 instr &= ~kImm26Mask;
1001 instr_at_put(pos, instr | (imm26 & kImm26Mask));
1005 void Assembler::print(
const Label* L) {
1006 if (L->is_unused()) {
1007 PrintF(
"unused label\n");
1008 }
else if (L->is_bound()) {
1009 PrintF(
"bound label to %d\n", L->pos());
1010 }
else if (L->is_linked()) {
1012 l.link_to(L->pos());
1013 PrintF(
"unbound label");
1014 while (l.is_linked()) {
1015 PrintF(
"@ %d ", l.pos());
1016 Instr instr = instr_at(l.pos());
1017 if ((instr & ~kImm16Mask) == 0) {
1020 PrintF(
"%d\n", instr);
1022 next(&l, is_internal_reference(&l));
1025 PrintF(
"label in inconsistent state (pos = %d)\n", L->pos_);
1030 void Assembler::bind_to(Label* L,
int pos) {
1031 DCHECK(0 <= pos && pos <= pc_offset());
1032 int trampoline_pos = kInvalidSlotPos;
1033 bool is_internal =
false;
1034 if (L->is_linked() && !trampoline_emitted_) {
1035 unbound_labels_count_--;
1036 if (!is_internal_reference(L)) {
1037 next_buffer_check_ += kTrampolineSlotsSize;
1041 while (L->is_linked()) {
1042 int fixup_pos = L->pos();
1043 int dist = pos - fixup_pos;
1044 is_internal = is_internal_reference(L);
1045 next(L, is_internal);
1047 Instr instr = instr_at(fixup_pos);
1049 target_at_put(fixup_pos, pos, is_internal);
1051 if (IsBranch(instr)) {
1052 int branch_offset = BranchOffset(instr);
1053 if (dist > branch_offset) {
1054 if (trampoline_pos == kInvalidSlotPos) {
1055 trampoline_pos = get_trampoline_entry(fixup_pos);
1056 CHECK_NE(trampoline_pos, kInvalidSlotPos);
1058 CHECK((trampoline_pos - fixup_pos) <= branch_offset);
1059 target_at_put(fixup_pos, trampoline_pos,
false);
1060 fixup_pos = trampoline_pos;
1062 target_at_put(fixup_pos, pos,
false);
1064 DCHECK(IsJ(instr) || IsJal(instr) || IsLui(instr) ||
1065 IsEmittedConstant(instr) || IsMov(instr, t8, ra));
1066 target_at_put(fixup_pos, pos,
false);
1074 if (pos > last_bound_pos_)
1075 last_bound_pos_ = pos;
1079 void Assembler::bind(Label* L) {
1080 DCHECK(!L->is_bound());
1081 bind_to(L, pc_offset());
1085 void Assembler::next(Label* L,
bool is_internal) {
1086 DCHECK(L->is_linked());
1087 int link = target_at(L->pos(), is_internal);
1088 if (link == kEndOfChain) {
1097 bool Assembler::is_near(Label* L) {
1098 DCHECK(L->is_bound());
1099 return pc_offset() - L->pos() < kMaxBranchOffset - 4 * kInstrSize;
1103 bool Assembler::is_near(Label* L, OffsetSize bits) {
1104 if (L ==
nullptr || !L->is_bound())
return true;
1105 return ((pc_offset() - L->pos()) <
1106 (1 << (bits + 2 - 1)) - 1 - 5 * kInstrSize);
1110 bool Assembler::is_near_branch(Label* L) {
1111 DCHECK(L->is_bound());
1112 return kArchVariant == kMips64r6 ? is_near_r6(L) : is_near_pre_r6(L);
1116 int Assembler::BranchOffset(Instr instr) {
1118 int bits = OffsetSize::kOffset16;
1120 if (kArchVariant == kMips64r6) {
1121 uint32_t opcode = GetOpcodeField(instr);
1126 bits = OffsetSize::kOffset26;
1132 if (GetRsField(instr) != 0) bits = OffsetSize::kOffset21;
1139 return (1 << (bits + 2 - 1)) - 1;
1147 bool Assembler::MustUseReg(RelocInfo::Mode rmode) {
1148 return !RelocInfo::IsNone(rmode);
1151 void Assembler::GenInstrRegister(Opcode opcode,
1156 SecondaryField func) {
1157 DCHECK(rd.is_valid() && rs.is_valid() && rt.is_valid() && is_uint5(sa));
1158 Instr instr = opcode | (rs.code() << kRsShift) | (rt.code() << kRtShift)
1159 | (rd.code() << kRdShift) | (sa << kSaShift) | func;
1164 void Assembler::GenInstrRegister(Opcode opcode,
1169 SecondaryField func) {
1170 DCHECK(rs.is_valid() && rt.is_valid() && is_uint5(msb) && is_uint5(lsb));
1171 Instr instr = opcode | (rs.code() << kRsShift) | (rt.code() << kRtShift)
1172 | (msb << kRdShift) | (lsb << kSaShift) | func;
1177 void Assembler::GenInstrRegister(Opcode opcode,
1182 SecondaryField func) {
1183 DCHECK(fd.is_valid() && fs.is_valid() && ft.is_valid());
1184 Instr instr = opcode | fmt | (ft.code() << kFtShift) | (fs.code() << kFsShift)
1185 | (fd.code() << kFdShift) | func;
1190 void Assembler::GenInstrRegister(Opcode opcode,
1195 SecondaryField func) {
1196 DCHECK(fd.is_valid() && fr.is_valid() && fs.is_valid() && ft.is_valid());
1197 Instr instr = opcode | (fr.code() << kFrShift) | (ft.code() << kFtShift)
1198 | (fs.code() << kFsShift) | (fd.code() << kFdShift) | func;
1203 void Assembler::GenInstrRegister(Opcode opcode,
1208 SecondaryField func) {
1209 DCHECK(fd.is_valid() && fs.is_valid() && rt.is_valid());
1210 Instr instr = opcode | fmt | (rt.code() << kRtShift)
1211 | (fs.code() << kFsShift) | (fd.code() << kFdShift) | func;
1216 void Assembler::GenInstrRegister(Opcode opcode,
1219 FPUControlRegister fs,
1220 SecondaryField func) {
1221 DCHECK(fs.is_valid() && rt.is_valid());
1223 opcode | fmt | (rt.code() << kRtShift) | (fs.code() << kFsShift) | func;
1230 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, Register rt,
1232 CompactBranchType is_compact_branch) {
1233 DCHECK(rs.is_valid() && rt.is_valid() && (is_int16(j) || is_uint16(j)));
1234 Instr instr = opcode | (rs.code() << kRsShift) | (rt.code() << kRtShift)
1236 emit(instr, is_compact_branch);
1239 void Assembler::GenInstrImmediate(Opcode opcode, Register base, Register rt,
1240 int32_t offset9,
int bit6,
1241 SecondaryField func) {
1242 DCHECK(base.is_valid() && rt.is_valid() && is_int9(offset9) &&
1244 Instr instr = opcode | (base.code() << kBaseShift) | (rt.code() << kRtShift) |
1245 ((offset9 << kImm9Shift) & kImm9Mask) | bit6 << kBit6Shift |
1250 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, SecondaryField SF,
1252 CompactBranchType is_compact_branch) {
1253 DCHECK(rs.is_valid() && (is_int16(j) || is_uint16(j)));
1254 Instr instr = opcode | (rs.code() << kRsShift) | SF | (j & kImm16Mask);
1255 emit(instr, is_compact_branch);
1259 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, FPURegister ft,
1261 CompactBranchType is_compact_branch) {
1262 DCHECK(rs.is_valid() && ft.is_valid() && (is_int16(j) || is_uint16(j)));
1263 Instr instr = opcode | (rs.code() << kRsShift) | (ft.code() << kFtShift)
1265 emit(instr, is_compact_branch);
1269 void Assembler::GenInstrImmediate(Opcode opcode, Register rs, int32_t offset21,
1270 CompactBranchType is_compact_branch) {
1271 DCHECK(rs.is_valid() && (is_int21(offset21)));
1272 Instr instr = opcode | (rs.code() << kRsShift) | (offset21 & kImm21Mask);
1273 emit(instr, is_compact_branch);
1277 void Assembler::GenInstrImmediate(Opcode opcode, Register rs,
1279 DCHECK(rs.is_valid() && (is_uint21(offset21)));
1280 Instr instr = opcode | (rs.code() << kRsShift) | (offset21 & kImm21Mask);
1285 void Assembler::GenInstrImmediate(Opcode opcode, int32_t offset26,
1286 CompactBranchType is_compact_branch) {
1287 DCHECK(is_int26(offset26));
1288 Instr instr = opcode | (offset26 & kImm26Mask);
1289 emit(instr, is_compact_branch);
1293 void Assembler::GenInstrJump(Opcode opcode,
1295 BlockTrampolinePoolScope block_trampoline_pool(
this);
1296 DCHECK(is_uint26(address));
1297 Instr instr = opcode | address;
1299 BlockTrampolinePoolFor(1);
1303 void Assembler::GenInstrMsaI8(SecondaryField operation,
uint32_t imm8,
1304 MSARegister ws, MSARegister wd) {
1305 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1306 DCHECK(ws.is_valid() && wd.is_valid() && is_uint8(imm8));
1307 Instr instr = MSA | operation | ((imm8 & kImm8Mask) << kWtShift) |
1308 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1312 void Assembler::GenInstrMsaI5(SecondaryField operation, SecondaryField df,
1313 int32_t imm5, MSARegister ws, MSARegister wd) {
1314 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1315 DCHECK(ws.is_valid() && wd.is_valid());
1316 DCHECK((operation == MAXI_S) || (operation == MINI_S) ||
1317 (operation == CEQI) || (operation == CLTI_S) ||
1318 (operation == CLEI_S)
1321 Instr instr = MSA | operation | df | ((imm5 & kImm5Mask) << kWtShift) |
1322 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1326 void Assembler::GenInstrMsaBit(SecondaryField operation, SecondaryField df,
1327 uint32_t m, MSARegister ws, MSARegister wd) {
1328 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1329 DCHECK(ws.is_valid() && wd.is_valid() && is_valid_msa_df_m(df, m));
1330 Instr instr = MSA | operation | df | (m << kWtShift) |
1331 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1335 void Assembler::GenInstrMsaI10(SecondaryField operation, SecondaryField df,
1336 int32_t imm10, MSARegister wd) {
1337 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1338 DCHECK(wd.is_valid() && is_int10(imm10));
1339 Instr instr = MSA | operation | df | ((imm10 & kImm10Mask) << kWsShift) |
1340 (wd.code() << kWdShift);
1344 template <
typename RegType>
1345 void Assembler::GenInstrMsa3R(SecondaryField operation, SecondaryField df,
1346 RegType t, MSARegister ws, MSARegister wd) {
1347 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1348 DCHECK(t.is_valid() && ws.is_valid() && wd.is_valid());
1349 Instr instr = MSA | operation | df | (t.code() << kWtShift) |
1350 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1354 template <
typename DstType,
typename SrcType>
1355 void Assembler::GenInstrMsaElm(SecondaryField operation, SecondaryField df,
1356 uint32_t n, SrcType src, DstType dst) {
1357 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1358 DCHECK(src.is_valid() && dst.is_valid() && is_valid_msa_df_n(df, n));
1359 Instr instr = MSA | operation | df | (n << kWtShift) |
1360 (src.code() << kWsShift) | (dst.code() << kWdShift) |
1365 void Assembler::GenInstrMsa3RF(SecondaryField operation,
uint32_t df,
1366 MSARegister wt, MSARegister ws, MSARegister wd) {
1367 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1368 DCHECK(wt.is_valid() && ws.is_valid() && wd.is_valid());
1370 Instr instr = MSA | operation | (df << 21) | (wt.code() << kWtShift) |
1371 (ws.code() << kWsShift) | (wd.code() << kWdShift);
1375 void Assembler::GenInstrMsaVec(SecondaryField operation, MSARegister wt,
1376 MSARegister ws, MSARegister wd) {
1377 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1378 DCHECK(wt.is_valid() && ws.is_valid() && wd.is_valid());
1379 Instr instr = MSA | operation | (wt.code() << kWtShift) |
1380 (ws.code() << kWsShift) | (wd.code() << kWdShift) |
1381 MSA_VEC_2R_2RF_MINOR;
1385 void Assembler::GenInstrMsaMI10(SecondaryField operation, int32_t s10,
1386 Register rs, MSARegister wd) {
1387 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1388 DCHECK(rs.is_valid() && wd.is_valid() && is_int10(s10));
1389 Instr instr = MSA | operation | ((s10 & kImm10Mask) << kWtShift) |
1390 (rs.code() << kWsShift) | (wd.code() << kWdShift);
1394 void Assembler::GenInstrMsa2R(SecondaryField operation, SecondaryField df,
1395 MSARegister ws, MSARegister wd) {
1396 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1397 DCHECK(ws.is_valid() && wd.is_valid());
1398 Instr instr = MSA | MSA_2R_FORMAT | operation | df | (ws.code() << kWsShift) |
1399 (wd.code() << kWdShift) | MSA_VEC_2R_2RF_MINOR;
1403 void Assembler::GenInstrMsa2RF(SecondaryField operation, SecondaryField df,
1404 MSARegister ws, MSARegister wd) {
1405 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1406 DCHECK(ws.is_valid() && wd.is_valid());
1407 Instr instr = MSA | MSA_2RF_FORMAT | operation | df |
1408 (ws.code() << kWsShift) | (wd.code() << kWdShift) |
1409 MSA_VEC_2R_2RF_MINOR;
1413 void Assembler::GenInstrMsaBranch(SecondaryField operation, MSARegister wt,
1415 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
1416 DCHECK(wt.is_valid() && is_int16(offset16));
1417 BlockTrampolinePoolScope block_trampoline_pool(
this);
1419 COP1 | operation | (wt.code() << kWtShift) | (offset16 & kImm16Mask);
1421 BlockTrampolinePoolFor(1);
1425 int32_t Assembler::get_trampoline_entry(int32_t pos) {
1426 int32_t trampoline_entry = kInvalidSlotPos;
1427 if (!internal_trampoline_exception_) {
1428 if (trampoline_.start() > pos) {
1429 trampoline_entry = trampoline_.take_slot();
1432 if (kInvalidSlotPos == trampoline_entry) {
1433 internal_trampoline_exception_ =
true;
1436 return trampoline_entry;
1440 uint64_t Assembler::jump_address(Label* L) {
1442 if (L->is_bound()) {
1443 target_pos = L->pos();
1445 if (L->is_linked()) {
1446 target_pos = L->pos();
1447 L->link_to(pc_offset());
1449 L->link_to(pc_offset());
1450 return kEndOfJumpChain;
1453 uint64_t imm =
reinterpret_cast<uint64_t
>(buffer_) + target_pos;
1454 DCHECK_EQ(imm & 3, 0);
1460 uint64_t Assembler::jump_offset(Label* L) {
1462 int32_t pad = IsPrevInstrCompactBranch() ? kInstrSize : 0;
1464 if (L->is_bound()) {
1465 target_pos = L->pos();
1467 if (L->is_linked()) {
1468 target_pos = L->pos();
1469 L->link_to(pc_offset() + pad);
1471 L->link_to(pc_offset() + pad);
1472 return kEndOfJumpChain;
1475 int64_t imm = target_pos - (pc_offset() + pad);
1476 DCHECK_EQ(imm & 3, 0);
1478 return static_cast<uint64_t
>(imm);
1481 uint64_t Assembler::branch_long_offset(Label* L) {
1484 if (L->is_bound()) {
1485 target_pos = L->pos();
1487 if (L->is_linked()) {
1488 target_pos = L->pos();
1489 L->link_to(pc_offset());
1491 L->link_to(pc_offset());
1492 return kEndOfJumpChain;
1495 int64_t offset = target_pos - (pc_offset() + kLongBranchPCOffset);
1496 DCHECK_EQ(offset & 3, 0);
1498 return static_cast<uint64_t
>(offset);
1501 int32_t Assembler::branch_offset_helper(Label* L, OffsetSize bits) {
1503 int32_t pad = IsPrevInstrCompactBranch() ? kInstrSize : 0;
1505 if (L->is_bound()) {
1506 target_pos = L->pos();
1508 if (L->is_linked()) {
1509 target_pos = L->pos();
1510 L->link_to(pc_offset() + pad);
1512 L->link_to(pc_offset() + pad);
1513 if (!trampoline_emitted_) {
1514 unbound_labels_count_++;
1515 next_buffer_check_ -= kTrampolineSlotsSize;
1521 int32_t offset = target_pos - (pc_offset() + kBranchPCOffset + pad);
1522 DCHECK(is_intn(offset, bits + 2));
1523 DCHECK_EQ(offset & 3, 0);
1529 void Assembler::label_at_put(Label* L,
int at_offset) {
1531 if (L->is_bound()) {
1532 target_pos = L->pos();
1533 instr_at_put(at_offset, target_pos + (Code::kHeaderSize - kHeapObjectTag));
1535 if (L->is_linked()) {
1536 target_pos = L->pos();
1537 int32_t imm18 = target_pos - at_offset;
1538 DCHECK_EQ(imm18 & 3, 0);
1539 int32_t imm16 = imm18 >> 2;
1540 DCHECK(is_int16(imm16));
1541 instr_at_put(at_offset, (imm16 & kImm16Mask));
1543 target_pos = kEndOfChain;
1544 instr_at_put(at_offset, 0);
1545 if (!trampoline_emitted_) {
1546 unbound_labels_count_++;
1547 next_buffer_check_ -= kTrampolineSlotsSize;
1550 L->link_to(at_offset);
1557 void Assembler::b(int16_t offset) {
1558 beq(zero_reg, zero_reg, offset);
1562 void Assembler::bal(int16_t offset) {
1563 bgezal(zero_reg, offset);
1567 void Assembler::bc(int32_t offset) {
1568 DCHECK_EQ(kArchVariant, kMips64r6);
1569 GenInstrImmediate(BC, offset, CompactBranchType::COMPACT_BRANCH);
1573 void Assembler::balc(int32_t offset) {
1574 DCHECK_EQ(kArchVariant, kMips64r6);
1575 GenInstrImmediate(BALC, offset, CompactBranchType::COMPACT_BRANCH);
1579 void Assembler::beq(Register rs, Register rt, int16_t offset) {
1580 BlockTrampolinePoolScope block_trampoline_pool(
this);
1581 GenInstrImmediate(BEQ, rs, rt, offset);
1582 BlockTrampolinePoolFor(1);
1586 void Assembler::bgez(Register rs, int16_t offset) {
1587 BlockTrampolinePoolScope block_trampoline_pool(
this);
1588 GenInstrImmediate(REGIMM, rs, BGEZ, offset);
1589 BlockTrampolinePoolFor(1);
1593 void Assembler::bgezc(Register rt, int16_t offset) {
1594 DCHECK_EQ(kArchVariant, kMips64r6);
1595 DCHECK(rt != zero_reg);
1596 GenInstrImmediate(BLEZL, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1600 void Assembler::bgeuc(Register rs, Register rt, int16_t offset) {
1601 DCHECK_EQ(kArchVariant, kMips64r6);
1602 DCHECK(rs != zero_reg);
1603 DCHECK(rt != zero_reg);
1604 DCHECK(rs.code() != rt.code());
1605 GenInstrImmediate(BLEZ, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1609 void Assembler::bgec(Register rs, Register rt, int16_t offset) {
1610 DCHECK_EQ(kArchVariant, kMips64r6);
1611 DCHECK(rs != zero_reg);
1612 DCHECK(rt != zero_reg);
1613 DCHECK(rs.code() != rt.code());
1614 GenInstrImmediate(BLEZL, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1618 void Assembler::bgezal(Register rs, int16_t offset) {
1619 DCHECK(kArchVariant != kMips64r6 || rs == zero_reg);
1621 BlockTrampolinePoolScope block_trampoline_pool(
this);
1622 GenInstrImmediate(REGIMM, rs, BGEZAL, offset);
1623 BlockTrampolinePoolFor(1);
1627 void Assembler::bgtz(Register rs, int16_t offset) {
1628 BlockTrampolinePoolScope block_trampoline_pool(
this);
1629 GenInstrImmediate(BGTZ, rs, zero_reg, offset);
1630 BlockTrampolinePoolFor(1);
1634 void Assembler::bgtzc(Register rt, int16_t offset) {
1635 DCHECK_EQ(kArchVariant, kMips64r6);
1636 DCHECK(rt != zero_reg);
1637 GenInstrImmediate(BGTZL, zero_reg, rt, offset,
1638 CompactBranchType::COMPACT_BRANCH);
1642 void Assembler::blez(Register rs, int16_t offset) {
1643 BlockTrampolinePoolScope block_trampoline_pool(
this);
1644 GenInstrImmediate(BLEZ, rs, zero_reg, offset);
1645 BlockTrampolinePoolFor(1);
1649 void Assembler::blezc(Register rt, int16_t offset) {
1650 DCHECK_EQ(kArchVariant, kMips64r6);
1651 DCHECK(rt != zero_reg);
1652 GenInstrImmediate(BLEZL, zero_reg, rt, offset,
1653 CompactBranchType::COMPACT_BRANCH);
1657 void Assembler::bltzc(Register rt, int16_t offset) {
1658 DCHECK_EQ(kArchVariant, kMips64r6);
1659 DCHECK(rt != zero_reg);
1660 GenInstrImmediate(BGTZL, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1664 void Assembler::bltuc(Register rs, Register rt, int16_t offset) {
1665 DCHECK_EQ(kArchVariant, kMips64r6);
1666 DCHECK(rs != zero_reg);
1667 DCHECK(rt != zero_reg);
1668 DCHECK(rs.code() != rt.code());
1669 GenInstrImmediate(BGTZ, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1673 void Assembler::bltc(Register rs, Register rt, int16_t offset) {
1674 DCHECK_EQ(kArchVariant, kMips64r6);
1675 DCHECK(rs != zero_reg);
1676 DCHECK(rt != zero_reg);
1677 DCHECK(rs.code() != rt.code());
1678 GenInstrImmediate(BGTZL, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1682 void Assembler::bltz(Register rs, int16_t offset) {
1683 BlockTrampolinePoolScope block_trampoline_pool(
this);
1684 GenInstrImmediate(REGIMM, rs, BLTZ, offset);
1685 BlockTrampolinePoolFor(1);
1689 void Assembler::bltzal(Register rs, int16_t offset) {
1690 DCHECK(kArchVariant != kMips64r6 || rs == zero_reg);
1692 BlockTrampolinePoolScope block_trampoline_pool(
this);
1693 GenInstrImmediate(REGIMM, rs, BLTZAL, offset);
1694 BlockTrampolinePoolFor(1);
1698 void Assembler::bne(Register rs, Register rt, int16_t offset) {
1699 BlockTrampolinePoolScope block_trampoline_pool(
this);
1700 GenInstrImmediate(BNE, rs, rt, offset);
1701 BlockTrampolinePoolFor(1);
1705 void Assembler::bovc(Register rs, Register rt, int16_t offset) {
1706 DCHECK_EQ(kArchVariant, kMips64r6);
1707 if (rs.code() >= rt.code()) {
1708 GenInstrImmediate(ADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1710 GenInstrImmediate(ADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1715 void Assembler::bnvc(Register rs, Register rt, int16_t offset) {
1716 DCHECK_EQ(kArchVariant, kMips64r6);
1717 if (rs.code() >= rt.code()) {
1718 GenInstrImmediate(DADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1720 GenInstrImmediate(DADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1725 void Assembler::blezalc(Register rt, int16_t offset) {
1726 DCHECK_EQ(kArchVariant, kMips64r6);
1727 DCHECK(rt != zero_reg);
1729 GenInstrImmediate(BLEZ, zero_reg, rt, offset,
1730 CompactBranchType::COMPACT_BRANCH);
1734 void Assembler::bgezalc(Register rt, int16_t offset) {
1735 DCHECK_EQ(kArchVariant, kMips64r6);
1736 DCHECK(rt != zero_reg);
1738 GenInstrImmediate(BLEZ, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1742 void Assembler::bgezall(Register rs, int16_t offset) {
1743 DCHECK_NE(kArchVariant, kMips64r6);
1744 DCHECK(rs != zero_reg);
1746 BlockTrampolinePoolScope block_trampoline_pool(
this);
1747 GenInstrImmediate(REGIMM, rs, BGEZALL, offset);
1748 BlockTrampolinePoolFor(1);
1752 void Assembler::bltzalc(Register rt, int16_t offset) {
1753 DCHECK_EQ(kArchVariant, kMips64r6);
1754 DCHECK(rt != zero_reg);
1756 GenInstrImmediate(BGTZ, rt, rt, offset, CompactBranchType::COMPACT_BRANCH);
1760 void Assembler::bgtzalc(Register rt, int16_t offset) {
1761 DCHECK_EQ(kArchVariant, kMips64r6);
1762 DCHECK(rt != zero_reg);
1764 GenInstrImmediate(BGTZ, zero_reg, rt, offset,
1765 CompactBranchType::COMPACT_BRANCH);
1769 void Assembler::beqzalc(Register rt, int16_t offset) {
1770 DCHECK_EQ(kArchVariant, kMips64r6);
1771 DCHECK(rt != zero_reg);
1773 GenInstrImmediate(ADDI, zero_reg, rt, offset,
1774 CompactBranchType::COMPACT_BRANCH);
1778 void Assembler::bnezalc(Register rt, int16_t offset) {
1779 DCHECK_EQ(kArchVariant, kMips64r6);
1780 DCHECK(rt != zero_reg);
1782 GenInstrImmediate(DADDI, zero_reg, rt, offset,
1783 CompactBranchType::COMPACT_BRANCH);
1787 void Assembler::beqc(Register rs, Register rt, int16_t offset) {
1788 DCHECK_EQ(kArchVariant, kMips64r6);
1789 DCHECK(rs.code() != rt.code() && rs.code() != 0 && rt.code() != 0);
1790 if (rs.code() < rt.code()) {
1791 GenInstrImmediate(ADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1793 GenInstrImmediate(ADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1798 void Assembler::beqzc(Register rs, int32_t offset) {
1799 DCHECK_EQ(kArchVariant, kMips64r6);
1800 DCHECK(rs != zero_reg);
1801 GenInstrImmediate(POP66, rs, offset, CompactBranchType::COMPACT_BRANCH);
1805 void Assembler::bnec(Register rs, Register rt, int16_t offset) {
1806 DCHECK_EQ(kArchVariant, kMips64r6);
1807 DCHECK(rs.code() != rt.code() && rs.code() != 0 && rt.code() != 0);
1808 if (rs.code() < rt.code()) {
1809 GenInstrImmediate(DADDI, rs, rt, offset, CompactBranchType::COMPACT_BRANCH);
1811 GenInstrImmediate(DADDI, rt, rs, offset, CompactBranchType::COMPACT_BRANCH);
1816 void Assembler::bnezc(Register rs, int32_t offset) {
1817 DCHECK_EQ(kArchVariant, kMips64r6);
1818 DCHECK(rs != zero_reg);
1819 GenInstrImmediate(POP76, rs, offset, CompactBranchType::COMPACT_BRANCH);
1823 void Assembler::j(
int64_t target) {
1829 void Assembler::j(Label* target) {
1835 void Assembler::jal(Label* target) {
1840 void Assembler::jal(
int64_t target) {
1846 void Assembler::jr(Register rs) {
1847 if (kArchVariant != kMips64r6) {
1848 BlockTrampolinePoolScope block_trampoline_pool(
this);
1849 GenInstrRegister(SPECIAL, rs, zero_reg, zero_reg, 0, JR);
1850 BlockTrampolinePoolFor(1);
1857 void Assembler::jalr(Register rs, Register rd) {
1858 DCHECK(rs.code() != rd.code());
1859 BlockTrampolinePoolScope block_trampoline_pool(
this);
1860 GenInstrRegister(SPECIAL, rs, zero_reg, rd, 0, JALR);
1861 BlockTrampolinePoolFor(1);
1865 void Assembler::jic(Register rt, int16_t offset) {
1866 DCHECK_EQ(kArchVariant, kMips64r6);
1867 GenInstrImmediate(POP66, zero_reg, rt, offset);
1871 void Assembler::jialc(Register rt, int16_t offset) {
1872 DCHECK_EQ(kArchVariant, kMips64r6);
1873 GenInstrImmediate(POP76, zero_reg, rt, offset);
1881 void Assembler::addu(Register rd, Register rs, Register rt) {
1882 GenInstrRegister(SPECIAL, rs, rt, rd, 0, ADDU);
1886 void Assembler::addiu(Register rd, Register rs, int32_t j) {
1887 GenInstrImmediate(ADDIU, rs, rd, j);
1891 void Assembler::subu(Register rd, Register rs, Register rt) {
1892 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SUBU);
1896 void Assembler::mul(Register rd, Register rs, Register rt) {
1897 if (kArchVariant == kMips64r6) {
1898 GenInstrRegister(SPECIAL, rs, rt, rd, MUL_OP, MUL_MUH);
1900 GenInstrRegister(SPECIAL2, rs, rt, rd, 0, MUL);
1905 void Assembler::muh(Register rd, Register rs, Register rt) {
1906 DCHECK_EQ(kArchVariant, kMips64r6);
1907 GenInstrRegister(SPECIAL, rs, rt, rd, MUH_OP, MUL_MUH);
1911 void Assembler::mulu(Register rd, Register rs, Register rt) {
1912 DCHECK_EQ(kArchVariant, kMips64r6);
1913 GenInstrRegister(SPECIAL, rs, rt, rd, MUL_OP, MUL_MUH_U);
1917 void Assembler::muhu(Register rd, Register rs, Register rt) {
1918 DCHECK_EQ(kArchVariant, kMips64r6);
1919 GenInstrRegister(SPECIAL, rs, rt, rd, MUH_OP, MUL_MUH_U);
1923 void Assembler::dmul(Register rd, Register rs, Register rt) {
1924 DCHECK_EQ(kArchVariant, kMips64r6);
1925 GenInstrRegister(SPECIAL, rs, rt, rd, MUL_OP, D_MUL_MUH);
1929 void Assembler::dmuh(Register rd, Register rs, Register rt) {
1930 DCHECK_EQ(kArchVariant, kMips64r6);
1931 GenInstrRegister(SPECIAL, rs, rt, rd, MUH_OP, D_MUL_MUH);
1935 void Assembler::dmulu(Register rd, Register rs, Register rt) {
1936 DCHECK_EQ(kArchVariant, kMips64r6);
1937 GenInstrRegister(SPECIAL, rs, rt, rd, MUL_OP, D_MUL_MUH_U);
1941 void Assembler::dmuhu(Register rd, Register rs, Register rt) {
1942 DCHECK_EQ(kArchVariant, kMips64r6);
1943 GenInstrRegister(SPECIAL, rs, rt, rd, MUH_OP, D_MUL_MUH_U);
1947 void Assembler::mult(Register rs, Register rt) {
1948 DCHECK_NE(kArchVariant, kMips64r6);
1949 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, MULT);
1953 void Assembler::multu(Register rs, Register rt) {
1954 DCHECK_NE(kArchVariant, kMips64r6);
1955 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, MULTU);
1959 void Assembler::daddiu(Register rd, Register rs, int32_t j) {
1960 GenInstrImmediate(DADDIU, rs, rd, j);
1964 void Assembler::div(Register rs, Register rt) {
1965 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DIV);
1969 void Assembler::div(Register rd, Register rs, Register rt) {
1970 DCHECK_EQ(kArchVariant, kMips64r6);
1971 GenInstrRegister(SPECIAL, rs, rt, rd, DIV_OP, DIV_MOD);
1975 void Assembler::mod(Register rd, Register rs, Register rt) {
1976 DCHECK_EQ(kArchVariant, kMips64r6);
1977 GenInstrRegister(SPECIAL, rs, rt, rd, MOD_OP, DIV_MOD);
1981 void Assembler::divu(Register rs, Register rt) {
1982 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DIVU);
1986 void Assembler::divu(Register rd, Register rs, Register rt) {
1987 DCHECK_EQ(kArchVariant, kMips64r6);
1988 GenInstrRegister(SPECIAL, rs, rt, rd, DIV_OP, DIV_MOD_U);
1992 void Assembler::modu(Register rd, Register rs, Register rt) {
1993 DCHECK_EQ(kArchVariant, kMips64r6);
1994 GenInstrRegister(SPECIAL, rs, rt, rd, MOD_OP, DIV_MOD_U);
1998 void Assembler::daddu(Register rd, Register rs, Register rt) {
1999 GenInstrRegister(SPECIAL, rs, rt, rd, 0, DADDU);
2003 void Assembler::dsubu(Register rd, Register rs, Register rt) {
2004 GenInstrRegister(SPECIAL, rs, rt, rd, 0, DSUBU);
2008 void Assembler::dmult(Register rs, Register rt) {
2009 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DMULT);
2013 void Assembler::dmultu(Register rs, Register rt) {
2014 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DMULTU);
2018 void Assembler::ddiv(Register rs, Register rt) {
2019 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DDIV);
2023 void Assembler::ddiv(Register rd, Register rs, Register rt) {
2024 DCHECK_EQ(kArchVariant, kMips64r6);
2025 GenInstrRegister(SPECIAL, rs, rt, rd, DIV_OP, D_DIV_MOD);
2029 void Assembler::dmod(Register rd, Register rs, Register rt) {
2030 DCHECK_EQ(kArchVariant, kMips64r6);
2031 GenInstrRegister(SPECIAL, rs, rt, rd, MOD_OP, D_DIV_MOD);
2035 void Assembler::ddivu(Register rs, Register rt) {
2036 GenInstrRegister(SPECIAL, rs, rt, zero_reg, 0, DDIVU);
2040 void Assembler::ddivu(Register rd, Register rs, Register rt) {
2041 DCHECK_EQ(kArchVariant, kMips64r6);
2042 GenInstrRegister(SPECIAL, rs, rt, rd, DIV_OP, D_DIV_MOD_U);
2046 void Assembler::dmodu(Register rd, Register rs, Register rt) {
2047 DCHECK_EQ(kArchVariant, kMips64r6);
2048 GenInstrRegister(SPECIAL, rs, rt, rd, MOD_OP, D_DIV_MOD_U);
2054 void Assembler::and_(Register rd, Register rs, Register rt) {
2055 GenInstrRegister(SPECIAL, rs, rt, rd, 0, AND);
2059 void Assembler::andi(Register rt, Register rs, int32_t j) {
2060 DCHECK(is_uint16(j));
2061 GenInstrImmediate(ANDI, rs, rt, j);
2065 void Assembler::or_(Register rd, Register rs, Register rt) {
2066 GenInstrRegister(SPECIAL, rs, rt, rd, 0, OR);
2070 void Assembler::ori(Register rt, Register rs, int32_t j) {
2071 DCHECK(is_uint16(j));
2072 GenInstrImmediate(ORI, rs, rt, j);
2076 void Assembler::xor_(Register rd, Register rs, Register rt) {
2077 GenInstrRegister(SPECIAL, rs, rt, rd, 0, XOR);
2081 void Assembler::xori(Register rt, Register rs, int32_t j) {
2082 DCHECK(is_uint16(j));
2083 GenInstrImmediate(XORI, rs, rt, j);
2087 void Assembler::nor(Register rd, Register rs, Register rt) {
2088 GenInstrRegister(SPECIAL, rs, rt, rd, 0, NOR);
2093 void Assembler::sll(Register rd,
2096 bool coming_from_nop) {
2100 DCHECK(coming_from_nop || (rd != zero_reg && rt != zero_reg));
2101 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, SLL);
2105 void Assembler::sllv(Register rd, Register rt, Register rs) {
2106 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SLLV);
2110 void Assembler::srl(Register rd, Register rt, uint16_t sa) {
2111 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, SRL);
2115 void Assembler::srlv(Register rd, Register rt, Register rs) {
2116 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SRLV);
2120 void Assembler::sra(Register rd, Register rt, uint16_t sa) {
2121 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, SRA);
2125 void Assembler::srav(Register rd, Register rt, Register rs) {
2126 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SRAV);
2130 void Assembler::rotr(Register rd, Register rt, uint16_t sa) {
2132 DCHECK(rd.is_valid() && rt.is_valid() && is_uint5(sa));
2133 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2134 Instr instr = SPECIAL | (1 << kRsShift) | (rt.code() << kRtShift)
2135 | (rd.code() << kRdShift) | (sa << kSaShift) | SRL;
2140 void Assembler::rotrv(Register rd, Register rt, Register rs) {
2142 DCHECK(rd.is_valid() && rt.is_valid() && rs.is_valid());
2143 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2144 Instr instr = SPECIAL | (rs.code() << kRsShift) | (rt.code() << kRtShift)
2145 | (rd.code() << kRdShift) | (1 << kSaShift) | SRLV;
2150 void Assembler::dsll(Register rd, Register rt, uint16_t sa) {
2151 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, DSLL);
2155 void Assembler::dsllv(Register rd, Register rt, Register rs) {
2156 GenInstrRegister(SPECIAL, rs, rt, rd, 0, DSLLV);
2160 void Assembler::dsrl(Register rd, Register rt, uint16_t sa) {
2161 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, DSRL);
2165 void Assembler::dsrlv(Register rd, Register rt, Register rs) {
2166 GenInstrRegister(SPECIAL, rs, rt, rd, 0, DSRLV);
2170 void Assembler::drotr(Register rd, Register rt, uint16_t sa) {
2171 DCHECK(rd.is_valid() && rt.is_valid() && is_uint5(sa));
2172 Instr instr = SPECIAL | (1 << kRsShift) | (rt.code() << kRtShift)
2173 | (rd.code() << kRdShift) | (sa << kSaShift) | DSRL;
2177 void Assembler::drotr32(Register rd, Register rt, uint16_t sa) {
2178 DCHECK(rd.is_valid() && rt.is_valid() && is_uint5(sa));
2179 Instr instr = SPECIAL | (1 << kRsShift) | (rt.code() << kRtShift) |
2180 (rd.code() << kRdShift) | (sa << kSaShift) | DSRL32;
2184 void Assembler::drotrv(Register rd, Register rt, Register rs) {
2185 DCHECK(rd.is_valid() && rt.is_valid() && rs.is_valid() );
2186 Instr instr = SPECIAL | (rs.code() << kRsShift) | (rt.code() << kRtShift)
2187 | (rd.code() << kRdShift) | (1 << kSaShift) | DSRLV;
2192 void Assembler::dsra(Register rd, Register rt, uint16_t sa) {
2193 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, DSRA);
2197 void Assembler::dsrav(Register rd, Register rt, Register rs) {
2198 GenInstrRegister(SPECIAL, rs, rt, rd, 0, DSRAV);
2202 void Assembler::dsll32(Register rd, Register rt, uint16_t sa) {
2203 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, DSLL32);
2207 void Assembler::dsrl32(Register rd, Register rt, uint16_t sa) {
2208 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, DSRL32);
2212 void Assembler::dsra32(Register rd, Register rt, uint16_t sa) {
2213 GenInstrRegister(SPECIAL, zero_reg, rt, rd, sa & 0x1F, DSRA32);
2217 void Assembler::lsa(Register rd, Register rt, Register rs, uint8_t sa) {
2218 DCHECK(rd.is_valid() && rt.is_valid() && rs.is_valid());
2220 DCHECK_EQ(kArchVariant, kMips64r6);
2221 Instr instr = SPECIAL | rs.code() << kRsShift | rt.code() << kRtShift |
2222 rd.code() << kRdShift | sa << kSaShift | LSA;
2227 void Assembler::dlsa(Register rd, Register rt, Register rs, uint8_t sa) {
2228 DCHECK(rd.is_valid() && rt.is_valid() && rs.is_valid());
2230 DCHECK_EQ(kArchVariant, kMips64r6);
2231 Instr instr = SPECIAL | rs.code() << kRsShift | rt.code() << kRtShift |
2232 rd.code() << kRdShift | sa << kSaShift | DLSA;
2239 void Assembler::AdjustBaseAndOffset(MemOperand& src,
2240 OffsetAccessType access_type,
2241 int second_access_add_to_offset) {
2252 bool doubleword_aligned = (src.offset() & (kDoubleSize - 1)) == 0;
2253 bool two_accesses =
static_cast<bool>(access_type) || !doubleword_aligned;
2254 DCHECK_LE(second_access_add_to_offset, 7);
2257 if (is_int16(src.offset()) &&
2258 (!two_accesses || is_int16(static_cast<int32_t>(
2259 src.offset() + second_access_add_to_offset)))) {
2270 uint32_t misalignment = src.offset() & (kDoubleSize - 1);
2278 constexpr int32_t kMinOffsetForSimpleAdjustment =
2280 constexpr int32_t kMaxOffsetForSimpleAdjustment =
2281 2 * kMinOffsetForSimpleAdjustment;
2283 UseScratchRegisterScope temps(
this);
2284 Register scratch = temps.Acquire();
2285 if (0 <= src.offset() && src.offset() <= kMaxOffsetForSimpleAdjustment) {
2286 daddiu(scratch, src.rm(), kMinOffsetForSimpleAdjustment);
2287 src.offset_ -= kMinOffsetForSimpleAdjustment;
2288 }
else if (-kMaxOffsetForSimpleAdjustment <= src.offset() &&
2290 daddiu(scratch, src.rm(), -kMinOffsetForSimpleAdjustment);
2291 src.offset_ += kMinOffsetForSimpleAdjustment;
2292 }
else if (kArchVariant == kMips64r6) {
2303 int16_t offset_low =
static_cast<uint16_t
>(src.offset());
2304 int32_t offset_low32 = offset_low;
2305 int16_t offset_high =
static_cast<uint16_t
>(src.offset() >> 16);
2306 bool increment_hi16 = offset_low < 0;
2307 bool overflow_hi16 =
false;
2309 if (increment_hi16) {
2311 overflow_hi16 = (offset_high == -32768);
2313 daui(scratch, src.rm(),
static_cast<uint16_t
>(offset_high));
2315 if (overflow_hi16) {
2319 if (two_accesses && !is_int16(static_cast<int32_t>(
2320 offset_low32 + second_access_add_to_offset))) {
2323 daddiu(scratch, scratch, kDoubleSize);
2324 offset_low32 -= kDoubleSize;
2327 src.offset_ = offset_low32;
2334 constexpr int32_t kMinOffsetForMediumAdjustment =
2335 2 * kMinOffsetForSimpleAdjustment;
2336 constexpr int32_t kMaxOffsetForMediumAdjustment =
2337 3 * kMinOffsetForSimpleAdjustment;
2338 if (0 <= src.offset() && src.offset() <= kMaxOffsetForMediumAdjustment) {
2339 daddiu(scratch, src.rm(), kMinOffsetForMediumAdjustment / 2);
2340 daddiu(scratch, scratch, kMinOffsetForMediumAdjustment / 2);
2341 src.offset_ -= kMinOffsetForMediumAdjustment;
2342 }
else if (-kMaxOffsetForMediumAdjustment <= src.offset() &&
2344 daddiu(scratch, src.rm(), -kMinOffsetForMediumAdjustment / 2);
2345 daddiu(scratch, scratch, -kMinOffsetForMediumAdjustment / 2);
2346 src.offset_ += kMinOffsetForMediumAdjustment;
2350 int32_t loaded_offset = RoundDown(src.offset(), kDoubleSize);
2351 lui(scratch, (loaded_offset >> kLuiShift) & kImm16Mask);
2352 ori(scratch, scratch, loaded_offset & kImm16Mask);
2353 daddu(scratch, scratch, src.rm());
2354 src.offset_ -= loaded_offset;
2359 DCHECK(is_int16(src.offset()));
2362 static_cast<int32_t>(src.offset() + second_access_add_to_offset)));
2364 DCHECK(misalignment == (src.offset() & (kDoubleSize - 1)));
2367 void Assembler::lb(Register rd,
const MemOperand& rs) {
2368 GenInstrImmediate(LB, rs.rm(), rd, rs.offset_);
2372 void Assembler::lbu(Register rd,
const MemOperand& rs) {
2373 GenInstrImmediate(LBU, rs.rm(), rd, rs.offset_);
2377 void Assembler::lh(Register rd,
const MemOperand& rs) {
2378 GenInstrImmediate(LH, rs.rm(), rd, rs.offset_);
2382 void Assembler::lhu(Register rd,
const MemOperand& rs) {
2383 GenInstrImmediate(LHU, rs.rm(), rd, rs.offset_);
2387 void Assembler::lw(Register rd,
const MemOperand& rs) {
2388 GenInstrImmediate(LW, rs.rm(), rd, rs.offset_);
2392 void Assembler::lwu(Register rd,
const MemOperand& rs) {
2393 GenInstrImmediate(LWU, rs.rm(), rd, rs.offset_);
2397 void Assembler::lwl(Register rd,
const MemOperand& rs) {
2398 DCHECK(is_int16(rs.offset_));
2399 DCHECK_EQ(kArchVariant, kMips64r2);
2400 GenInstrImmediate(LWL, rs.rm(), rd, rs.offset_);
2404 void Assembler::lwr(Register rd,
const MemOperand& rs) {
2405 DCHECK(is_int16(rs.offset_));
2406 DCHECK_EQ(kArchVariant, kMips64r2);
2407 GenInstrImmediate(LWR, rs.rm(), rd, rs.offset_);
2411 void Assembler::sb(Register rd,
const MemOperand& rs) {
2412 GenInstrImmediate(SB, rs.rm(), rd, rs.offset_);
2416 void Assembler::sh(Register rd,
const MemOperand& rs) {
2417 GenInstrImmediate(SH, rs.rm(), rd, rs.offset_);
2421 void Assembler::sw(Register rd,
const MemOperand& rs) {
2422 GenInstrImmediate(SW, rs.rm(), rd, rs.offset_);
2426 void Assembler::swl(Register rd,
const MemOperand& rs) {
2427 DCHECK(is_int16(rs.offset_));
2428 DCHECK_EQ(kArchVariant, kMips64r2);
2429 GenInstrImmediate(SWL, rs.rm(), rd, rs.offset_);
2433 void Assembler::swr(Register rd,
const MemOperand& rs) {
2434 DCHECK(is_int16(rs.offset_));
2435 DCHECK_EQ(kArchVariant, kMips64r2);
2436 GenInstrImmediate(SWR, rs.rm(), rd, rs.offset_);
2439 void Assembler::ll(Register rd,
const MemOperand& rs) {
2440 if (kArchVariant == kMips64r6) {
2441 DCHECK(is_int9(rs.offset_));
2442 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 0, LL_R6);
2444 DCHECK_EQ(kArchVariant, kMips64r2);
2445 DCHECK(is_int16(rs.offset_));
2446 GenInstrImmediate(LL, rs.rm(), rd, rs.offset_);
2450 void Assembler::lld(Register rd,
const MemOperand& rs) {
2451 if (kArchVariant == kMips64r6) {
2452 DCHECK(is_int9(rs.offset_));
2453 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 0, LLD_R6);
2455 DCHECK_EQ(kArchVariant, kMips64r2);
2456 DCHECK(is_int16(rs.offset_));
2457 GenInstrImmediate(LLD, rs.rm(), rd, rs.offset_);
2461 void Assembler::sc(Register rd,
const MemOperand& rs) {
2462 if (kArchVariant == kMips64r6) {
2463 DCHECK(is_int9(rs.offset_));
2464 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 0, SC_R6);
2466 DCHECK_EQ(kArchVariant, kMips64r2);
2467 GenInstrImmediate(SC, rs.rm(), rd, rs.offset_);
2471 void Assembler::scd(Register rd,
const MemOperand& rs) {
2472 if (kArchVariant == kMips64r6) {
2473 DCHECK(is_int9(rs.offset_));
2474 GenInstrImmediate(SPECIAL3, rs.rm(), rd, rs.offset_, 0, SCD_R6);
2476 DCHECK_EQ(kArchVariant, kMips64r2);
2477 GenInstrImmediate(SCD, rs.rm(), rd, rs.offset_);
2481 void Assembler::lui(Register rd, int32_t j) {
2482 DCHECK(is_uint16(j) || is_int16(j));
2483 GenInstrImmediate(LUI, zero_reg, rd, j);
2487 void Assembler::aui(Register rt, Register rs, int32_t j) {
2490 DCHECK(is_uint16(j));
2491 GenInstrImmediate(LUI, rs, rt, j);
2495 void Assembler::daui(Register rt, Register rs, int32_t j) {
2496 DCHECK(is_uint16(j));
2497 DCHECK(rs != zero_reg);
2498 GenInstrImmediate(DAUI, rs, rt, j);
2502 void Assembler::dahi(Register rs, int32_t j) {
2503 DCHECK(is_uint16(j));
2504 GenInstrImmediate(REGIMM, rs, DAHI, j);
2508 void Assembler::dati(Register rs, int32_t j) {
2509 DCHECK(is_uint16(j));
2510 GenInstrImmediate(REGIMM, rs, DATI, j);
2514 void Assembler::ldl(Register rd,
const MemOperand& rs) {
2515 DCHECK(is_int16(rs.offset_));
2516 DCHECK_EQ(kArchVariant, kMips64r2);
2517 GenInstrImmediate(LDL, rs.rm(), rd, rs.offset_);
2521 void Assembler::ldr(Register rd,
const MemOperand& rs) {
2522 DCHECK(is_int16(rs.offset_));
2523 DCHECK_EQ(kArchVariant, kMips64r2);
2524 GenInstrImmediate(LDR, rs.rm(), rd, rs.offset_);
2528 void Assembler::sdl(Register rd,
const MemOperand& rs) {
2529 DCHECK(is_int16(rs.offset_));
2530 DCHECK_EQ(kArchVariant, kMips64r2);
2531 GenInstrImmediate(SDL, rs.rm(), rd, rs.offset_);
2535 void Assembler::sdr(Register rd,
const MemOperand& rs) {
2536 DCHECK(is_int16(rs.offset_));
2537 DCHECK_EQ(kArchVariant, kMips64r2);
2538 GenInstrImmediate(SDR, rs.rm(), rd, rs.offset_);
2542 void Assembler::ld(Register rd,
const MemOperand& rs) {
2543 GenInstrImmediate(LD, rs.rm(), rd, rs.offset_);
2547 void Assembler::sd(Register rd,
const MemOperand& rs) {
2548 GenInstrImmediate(SD, rs.rm(), rd, rs.offset_);
2554 void Assembler::addiupc(Register rs, int32_t imm19) {
2555 DCHECK_EQ(kArchVariant, kMips64r6);
2556 DCHECK(rs.is_valid() && is_int19(imm19));
2557 uint32_t imm21 = ADDIUPC << kImm19Bits | (imm19 & kImm19Mask);
2558 GenInstrImmediate(PCREL, rs, imm21);
2562 void Assembler::lwpc(Register rs, int32_t offset19) {
2563 DCHECK_EQ(kArchVariant, kMips64r6);
2564 DCHECK(rs.is_valid() && is_int19(offset19));
2565 uint32_t imm21 = LWPC << kImm19Bits | (offset19 & kImm19Mask);
2566 GenInstrImmediate(PCREL, rs, imm21);
2570 void Assembler::lwupc(Register rs, int32_t offset19) {
2571 DCHECK_EQ(kArchVariant, kMips64r6);
2572 DCHECK(rs.is_valid() && is_int19(offset19));
2573 uint32_t imm21 = LWUPC << kImm19Bits | (offset19 & kImm19Mask);
2574 GenInstrImmediate(PCREL, rs, imm21);
2578 void Assembler::ldpc(Register rs, int32_t offset18) {
2579 DCHECK_EQ(kArchVariant, kMips64r6);
2580 DCHECK(rs.is_valid() && is_int18(offset18));
2581 uint32_t imm21 = LDPC << kImm18Bits | (offset18 & kImm18Mask);
2582 GenInstrImmediate(PCREL, rs, imm21);
2586 void Assembler::auipc(Register rs, int16_t imm16) {
2587 DCHECK_EQ(kArchVariant, kMips64r6);
2588 DCHECK(rs.is_valid());
2589 uint32_t imm21 = AUIPC << kImm16Bits | (imm16 & kImm16Mask);
2590 GenInstrImmediate(PCREL, rs, imm21);
2594 void Assembler::aluipc(Register rs, int16_t imm16) {
2595 DCHECK_EQ(kArchVariant, kMips64r6);
2596 DCHECK(rs.is_valid());
2597 uint32_t imm21 = ALUIPC << kImm16Bits | (imm16 & kImm16Mask);
2598 GenInstrImmediate(PCREL, rs, imm21);
2605 void Assembler::break_(
uint32_t code,
bool break_as_stop) {
2606 DCHECK_EQ(code & ~0xFFFFF, 0);
2610 DCHECK((break_as_stop &&
2611 code <= kMaxStopCode &&
2612 code > kMaxWatchpointCode) ||
2614 (code > kMaxStopCode ||
2615 code <= kMaxWatchpointCode)));
2616 Instr break_instr = SPECIAL | BREAK | (code << 6);
2621 void Assembler::stop(
const char* msg,
uint32_t code) {
2622 DCHECK_GT(code, kMaxWatchpointCode);
2623 DCHECK_LE(code, kMaxStopCode);
2624 #if defined(V8_HOST_ARCH_MIPS) || defined(V8_HOST_ARCH_MIPS64) 2626 #else // V8_HOST_ARCH_MIPS 2632 void Assembler::tge(Register rs, Register rt, uint16_t code) {
2633 DCHECK(is_uint10(code));
2634 Instr instr = SPECIAL | TGE | rs.code() << kRsShift
2635 | rt.code() << kRtShift | code << 6;
2640 void Assembler::tgeu(Register rs, Register rt, uint16_t code) {
2641 DCHECK(is_uint10(code));
2642 Instr instr = SPECIAL | TGEU | rs.code() << kRsShift
2643 | rt.code() << kRtShift | code << 6;
2648 void Assembler::tlt(Register rs, Register rt, uint16_t code) {
2649 DCHECK(is_uint10(code));
2651 SPECIAL | TLT | rs.code() << kRsShift | rt.code() << kRtShift | code << 6;
2656 void Assembler::tltu(Register rs, Register rt, uint16_t code) {
2657 DCHECK(is_uint10(code));
2659 SPECIAL | TLTU | rs.code() << kRsShift
2660 | rt.code() << kRtShift | code << 6;
2665 void Assembler::teq(Register rs, Register rt, uint16_t code) {
2666 DCHECK(is_uint10(code));
2668 SPECIAL | TEQ | rs.code() << kRsShift | rt.code() << kRtShift | code << 6;
2673 void Assembler::tne(Register rs, Register rt, uint16_t code) {
2674 DCHECK(is_uint10(code));
2676 SPECIAL | TNE | rs.code() << kRsShift | rt.code() << kRtShift | code << 6;
2680 void Assembler::sync() {
2681 Instr sync_instr = SPECIAL | SYNC;
2687 void Assembler::mfhi(Register rd) {
2688 GenInstrRegister(SPECIAL, zero_reg, zero_reg, rd, 0, MFHI);
2692 void Assembler::mflo(Register rd) {
2693 GenInstrRegister(SPECIAL, zero_reg, zero_reg, rd, 0, MFLO);
2698 void Assembler::slt(Register rd, Register rs, Register rt) {
2699 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SLT);
2703 void Assembler::sltu(Register rd, Register rs, Register rt) {
2704 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SLTU);
2708 void Assembler::slti(Register rt, Register rs, int32_t j) {
2709 GenInstrImmediate(SLTI, rs, rt, j);
2713 void Assembler::sltiu(Register rt, Register rs, int32_t j) {
2714 GenInstrImmediate(SLTIU, rs, rt, j);
2719 void Assembler::movz(Register rd, Register rs, Register rt) {
2720 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVZ);
2724 void Assembler::movn(Register rd, Register rs, Register rt) {
2725 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVN);
2729 void Assembler::movt(Register rd, Register rs, uint16_t cc) {
2730 Register rt = Register::from_code((cc & 0x0007) << 2 | 1);
2731 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVCI);
2735 void Assembler::movf(Register rd, Register rs, uint16_t cc) {
2736 Register rt = Register::from_code((cc & 0x0007) << 2 | 0);
2737 GenInstrRegister(SPECIAL, rs, rt, rd, 0, MOVCI);
2741 void Assembler::min_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2746 void Assembler::min_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2751 void Assembler::max_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2756 void Assembler::max_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2761 void Assembler::mina_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2762 mina(S, fd, fs, ft);
2766 void Assembler::mina_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2767 mina(D, fd, fs, ft);
2771 void Assembler::maxa_s(FPURegister fd, FPURegister fs, FPURegister ft) {
2772 maxa(S, fd, fs, ft);
2776 void Assembler::maxa_d(FPURegister fd, FPURegister fs, FPURegister ft) {
2777 maxa(D, fd, fs, ft);
2781 void Assembler::max(SecondaryField fmt, FPURegister fd, FPURegister fs,
2783 DCHECK_EQ(kArchVariant, kMips64r6);
2784 DCHECK((fmt == D) || (fmt == S));
2785 GenInstrRegister(COP1, fmt, ft, fs, fd, MAX);
2789 void Assembler::min(SecondaryField fmt, FPURegister fd, FPURegister fs,
2791 DCHECK_EQ(kArchVariant, kMips64r6);
2792 DCHECK((fmt == D) || (fmt == S));
2793 GenInstrRegister(COP1, fmt, ft, fs, fd, MIN);
2798 void Assembler::seleqz(Register rd, Register rs, Register rt) {
2799 DCHECK_EQ(kArchVariant, kMips64r6);
2800 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SELEQZ_S);
2805 void Assembler::selnez(Register rd, Register rs, Register rt) {
2806 DCHECK_EQ(kArchVariant, kMips64r6);
2807 GenInstrRegister(SPECIAL, rs, rt, rd, 0, SELNEZ_S);
2812 void Assembler::clz(Register rd, Register rs) {
2813 if (kArchVariant != kMips64r6) {
2815 GenInstrRegister(SPECIAL2, rs, rd, rd, 0, CLZ);
2817 GenInstrRegister(SPECIAL, rs, zero_reg, rd, 1, CLZ_R6);
2822 void Assembler::dclz(Register rd, Register rs) {
2823 if (kArchVariant != kMips64r6) {
2825 GenInstrRegister(SPECIAL2, rs, rd, rd, 0, DCLZ);
2827 GenInstrRegister(SPECIAL, rs, zero_reg, rd, 1, DCLZ_R6);
2832 void Assembler::ins_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2835 DCHECK((kArchVariant == kMips64r2) || (kArchVariant == kMips64r6));
2836 GenInstrRegister(SPECIAL3, rs, rt, pos + size - 1, pos, INS);
2840 void Assembler::dins_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2843 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2844 GenInstrRegister(SPECIAL3, rs, rt, pos + size - 1, pos, DINS);
2847 void Assembler::dinsm_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2850 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2851 GenInstrRegister(SPECIAL3, rs, rt, pos + size - 1 - 32, pos, DINSM);
2854 void Assembler::dinsu_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2857 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2858 GenInstrRegister(SPECIAL3, rs, rt, pos + size - 1 - 32, pos - 32, DINSU);
2861 void Assembler::ext_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2864 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2865 GenInstrRegister(SPECIAL3, rs, rt, size - 1, pos, EXT);
2869 void Assembler::dext_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2872 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2873 GenInstrRegister(SPECIAL3, rs, rt, size - 1, pos, DEXT);
2876 void Assembler::dextm_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2879 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2880 GenInstrRegister(SPECIAL3, rs, rt, size - 1 - 32, pos, DEXTM);
2883 void Assembler::dextu_(Register rt, Register rs, uint16_t pos, uint16_t size) {
2886 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2887 GenInstrRegister(SPECIAL3, rs, rt, size - 1, pos - 32, DEXTU);
2891 void Assembler::bitswap(Register rd, Register rt) {
2892 DCHECK_EQ(kArchVariant, kMips64r6);
2893 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, 0, BSHFL);
2897 void Assembler::dbitswap(Register rd, Register rt) {
2898 DCHECK_EQ(kArchVariant, kMips64r6);
2899 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, 0, DBSHFL);
2903 void Assembler::pref(int32_t hint,
const MemOperand& rs) {
2904 DCHECK(is_uint5(hint) && is_uint16(rs.offset_));
2905 Instr instr = PREF | (rs.rm().code() << kRsShift) | (hint << kRtShift)
2911 void Assembler::align(Register rd, Register rs, Register rt, uint8_t bp) {
2912 DCHECK_EQ(kArchVariant, kMips64r6);
2913 DCHECK(is_uint3(bp));
2914 uint16_t sa = (ALIGN << kBp2Bits) | bp;
2915 GenInstrRegister(SPECIAL3, rs, rt, rd, sa, BSHFL);
2919 void Assembler::dalign(Register rd, Register rs, Register rt, uint8_t bp) {
2920 DCHECK_EQ(kArchVariant, kMips64r6);
2921 DCHECK(is_uint3(bp));
2922 uint16_t sa = (DALIGN << kBp3Bits) | bp;
2923 GenInstrRegister(SPECIAL3, rs, rt, rd, sa, DBSHFL);
2926 void Assembler::wsbh(Register rd, Register rt) {
2927 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2928 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, WSBH, BSHFL);
2931 void Assembler::dsbh(Register rd, Register rt) {
2932 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2933 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, DSBH, DBSHFL);
2936 void Assembler::dshd(Register rd, Register rt) {
2937 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2938 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, DSHD, DBSHFL);
2941 void Assembler::seh(Register rd, Register rt) {
2942 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2943 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, SEH, BSHFL);
2946 void Assembler::seb(Register rd, Register rt) {
2947 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
2948 GenInstrRegister(SPECIAL3, zero_reg, rt, rd, SEB, BSHFL);
2954 void Assembler::lwc1(FPURegister fd,
const MemOperand& src) {
2955 GenInstrImmediate(LWC1, src.rm(), fd, src.offset_);
2959 void Assembler::ldc1(FPURegister fd,
const MemOperand& src) {
2960 GenInstrImmediate(LDC1, src.rm(), fd, src.offset_);
2963 void Assembler::swc1(FPURegister fs,
const MemOperand& src) {
2964 GenInstrImmediate(SWC1, src.rm(), fs, src.offset_);
2967 void Assembler::sdc1(FPURegister fs,
const MemOperand& src) {
2968 GenInstrImmediate(SDC1, src.rm(), fs, src.offset_);
2972 void Assembler::mtc1(Register rt, FPURegister fs) {
2973 GenInstrRegister(COP1, MTC1, rt, fs, f0);
2977 void Assembler::mthc1(Register rt, FPURegister fs) {
2978 GenInstrRegister(COP1, MTHC1, rt, fs, f0);
2982 void Assembler::dmtc1(Register rt, FPURegister fs) {
2983 GenInstrRegister(COP1, DMTC1, rt, fs, f0);
2987 void Assembler::mfc1(Register rt, FPURegister fs) {
2988 GenInstrRegister(COP1, MFC1, rt, fs, f0);
2992 void Assembler::mfhc1(Register rt, FPURegister fs) {
2993 GenInstrRegister(COP1, MFHC1, rt, fs, f0);
2997 void Assembler::dmfc1(Register rt, FPURegister fs) {
2998 GenInstrRegister(COP1, DMFC1, rt, fs, f0);
3002 void Assembler::ctc1(Register rt, FPUControlRegister fs) {
3003 GenInstrRegister(COP1, CTC1, rt, fs);
3007 void Assembler::cfc1(Register rt, FPUControlRegister fs) {
3008 GenInstrRegister(COP1, CFC1, rt, fs);
3012 void Assembler::sel(SecondaryField fmt, FPURegister fd, FPURegister fs,
3014 DCHECK_EQ(kArchVariant, kMips64r6);
3015 DCHECK((fmt == D) || (fmt == S));
3017 GenInstrRegister(COP1, fmt, ft, fs, fd,
SEL);
3021 void Assembler::sel_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3026 void Assembler::sel_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3032 void Assembler::seleqz(SecondaryField fmt, FPURegister fd, FPURegister fs,
3034 DCHECK((fmt == D) || (fmt == S));
3035 GenInstrRegister(COP1, fmt, ft, fs, fd, SELEQZ_C);
3039 void Assembler::seleqz_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3040 seleqz(D, fd, fs, ft);
3044 void Assembler::seleqz_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3045 seleqz(S, fd, fs, ft);
3049 void Assembler::selnez_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3050 selnez(D, fd, fs, ft);
3054 void Assembler::selnez_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3055 selnez(S, fd, fs, ft);
3059 void Assembler::movz_s(FPURegister fd, FPURegister fs, Register rt) {
3060 DCHECK_EQ(kArchVariant, kMips64r2);
3061 GenInstrRegister(COP1, S, rt, fs, fd, MOVZ_C);
3065 void Assembler::movz_d(FPURegister fd, FPURegister fs, Register rt) {
3066 DCHECK_EQ(kArchVariant, kMips64r2);
3067 GenInstrRegister(COP1, D, rt, fs, fd, MOVZ_C);
3071 void Assembler::movt_s(FPURegister fd, FPURegister fs, uint16_t cc) {
3072 DCHECK_EQ(kArchVariant, kMips64r2);
3073 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 1);
3074 GenInstrRegister(COP1, S, ft, fs, fd, MOVF);
3078 void Assembler::movt_d(FPURegister fd, FPURegister fs, uint16_t cc) {
3079 DCHECK_EQ(kArchVariant, kMips64r2);
3080 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 1);
3081 GenInstrRegister(COP1, D, ft, fs, fd, MOVF);
3085 void Assembler::movf_s(FPURegister fd, FPURegister fs, uint16_t cc) {
3086 DCHECK_EQ(kArchVariant, kMips64r2);
3087 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 0);
3088 GenInstrRegister(COP1, S, ft, fs, fd, MOVF);
3092 void Assembler::movf_d(FPURegister fd, FPURegister fs, uint16_t cc) {
3093 DCHECK_EQ(kArchVariant, kMips64r2);
3094 FPURegister ft = FPURegister::from_code((cc & 0x0007) << 2 | 0);
3095 GenInstrRegister(COP1, D, ft, fs, fd, MOVF);
3099 void Assembler::movn_s(FPURegister fd, FPURegister fs, Register rt) {
3100 DCHECK_EQ(kArchVariant, kMips64r2);
3101 GenInstrRegister(COP1, S, rt, fs, fd, MOVN_C);
3105 void Assembler::movn_d(FPURegister fd, FPURegister fs, Register rt) {
3106 DCHECK_EQ(kArchVariant, kMips64r2);
3107 GenInstrRegister(COP1, D, rt, fs, fd, MOVN_C);
3112 void Assembler::selnez(SecondaryField fmt, FPURegister fd, FPURegister fs,
3114 DCHECK_EQ(kArchVariant, kMips64r6);
3115 DCHECK((fmt == D) || (fmt == S));
3116 GenInstrRegister(COP1, fmt, ft, fs, fd, SELNEZ_C);
3122 void Assembler::add_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3123 GenInstrRegister(COP1, S, ft, fs, fd, ADD_D);
3127 void Assembler::add_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3128 GenInstrRegister(COP1, D, ft, fs, fd, ADD_D);
3132 void Assembler::sub_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3133 GenInstrRegister(COP1, S, ft, fs, fd, SUB_D);
3137 void Assembler::sub_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3138 GenInstrRegister(COP1, D, ft, fs, fd, SUB_D);
3142 void Assembler::mul_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3143 GenInstrRegister(COP1, S, ft, fs, fd, MUL_D);
3147 void Assembler::mul_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3148 GenInstrRegister(COP1, D, ft, fs, fd, MUL_D);
3151 void Assembler::madd_s(FPURegister fd, FPURegister fr, FPURegister fs,
3159 void Assembler::madd_d(FPURegister fd, FPURegister fr, FPURegister fs,
3167 void Assembler::msub_s(FPURegister fd, FPURegister fr, FPURegister fs,
3173 void Assembler::msub_d(FPURegister fd, FPURegister fr, FPURegister fs,
3179 void Assembler::maddf_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3180 DCHECK_EQ(kArchVariant, kMips64r6);
3181 GenInstrRegister(COP1, S, ft, fs, fd, MADDF_S);
3184 void Assembler::maddf_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3185 DCHECK_EQ(kArchVariant, kMips64r6);
3186 GenInstrRegister(COP1, D, ft, fs, fd, MADDF_D);
3189 void Assembler::msubf_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3190 DCHECK_EQ(kArchVariant, kMips64r6);
3191 GenInstrRegister(COP1, S, ft, fs, fd, MSUBF_S);
3194 void Assembler::msubf_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3195 DCHECK_EQ(kArchVariant, kMips64r6);
3196 GenInstrRegister(COP1, D, ft, fs, fd, MSUBF_D);
3199 void Assembler::div_s(FPURegister fd, FPURegister fs, FPURegister ft) {
3200 GenInstrRegister(COP1, S, ft, fs, fd, DIV_D);
3204 void Assembler::div_d(FPURegister fd, FPURegister fs, FPURegister ft) {
3205 GenInstrRegister(COP1, D, ft, fs, fd, DIV_D);
3209 void Assembler::abs_s(FPURegister fd, FPURegister fs) {
3210 GenInstrRegister(COP1, S, f0, fs, fd, ABS_D);
3214 void Assembler::abs_d(FPURegister fd, FPURegister fs) {
3215 GenInstrRegister(COP1, D, f0, fs, fd, ABS_D);
3219 void Assembler::mov_d(FPURegister fd, FPURegister fs) {
3220 GenInstrRegister(COP1, D, f0, fs, fd, MOV_D);
3224 void Assembler::mov_s(FPURegister fd, FPURegister fs) {
3225 GenInstrRegister(COP1, S, f0, fs, fd, MOV_S);
3229 void Assembler::neg_s(FPURegister fd, FPURegister fs) {
3230 GenInstrRegister(COP1, S, f0, fs, fd, NEG_D);
3234 void Assembler::neg_d(FPURegister fd, FPURegister fs) {
3235 GenInstrRegister(COP1, D, f0, fs, fd, NEG_D);
3239 void Assembler::sqrt_s(FPURegister fd, FPURegister fs) {
3240 GenInstrRegister(COP1, S, f0, fs, fd, SQRT_D);
3244 void Assembler::sqrt_d(FPURegister fd, FPURegister fs) {
3245 GenInstrRegister(COP1, D, f0, fs, fd, SQRT_D);
3249 void Assembler::rsqrt_s(FPURegister fd, FPURegister fs) {
3250 GenInstrRegister(COP1, S, f0, fs, fd, RSQRT_S);
3254 void Assembler::rsqrt_d(FPURegister fd, FPURegister fs) {
3255 GenInstrRegister(COP1, D, f0, fs, fd, RSQRT_D);
3259 void Assembler::recip_d(FPURegister fd, FPURegister fs) {
3260 GenInstrRegister(COP1, D, f0, fs, fd, RECIP_D);
3264 void Assembler::recip_s(FPURegister fd, FPURegister fs) {
3265 GenInstrRegister(COP1, S, f0, fs, fd, RECIP_S);
3270 void Assembler::cvt_w_s(FPURegister fd, FPURegister fs) {
3271 GenInstrRegister(COP1, S, f0, fs, fd, CVT_W_S);
3275 void Assembler::cvt_w_d(FPURegister fd, FPURegister fs) {
3276 GenInstrRegister(COP1, D, f0, fs, fd, CVT_W_D);
3280 void Assembler::trunc_w_s(FPURegister fd, FPURegister fs) {
3281 GenInstrRegister(COP1, S, f0, fs, fd, TRUNC_W_S);
3285 void Assembler::trunc_w_d(FPURegister fd, FPURegister fs) {
3286 GenInstrRegister(COP1, D, f0, fs, fd, TRUNC_W_D);
3290 void Assembler::round_w_s(FPURegister fd, FPURegister fs) {
3291 GenInstrRegister(COP1, S, f0, fs, fd, ROUND_W_S);
3295 void Assembler::round_w_d(FPURegister fd, FPURegister fs) {
3296 GenInstrRegister(COP1, D, f0, fs, fd, ROUND_W_D);
3300 void Assembler::floor_w_s(FPURegister fd, FPURegister fs) {
3301 GenInstrRegister(COP1, S, f0, fs, fd, FLOOR_W_S);
3305 void Assembler::floor_w_d(FPURegister fd, FPURegister fs) {
3306 GenInstrRegister(COP1, D, f0, fs, fd, FLOOR_W_D);
3310 void Assembler::ceil_w_s(FPURegister fd, FPURegister fs) {
3311 GenInstrRegister(COP1, S, f0, fs, fd, CEIL_W_S);
3315 void Assembler::ceil_w_d(FPURegister fd, FPURegister fs) {
3316 GenInstrRegister(COP1, D, f0, fs, fd, CEIL_W_D);
3320 void Assembler::rint_s(FPURegister fd, FPURegister fs) { rint(S, fd, fs); }
3323 void Assembler::rint_d(FPURegister fd, FPURegister fs) { rint(D, fd, fs); }
3326 void Assembler::rint(SecondaryField fmt, FPURegister fd, FPURegister fs) {
3327 DCHECK_EQ(kArchVariant, kMips64r6);
3328 GenInstrRegister(COP1, fmt, f0, fs, fd, RINT);
3332 void Assembler::cvt_l_s(FPURegister fd, FPURegister fs) {
3333 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
3334 GenInstrRegister(COP1, S, f0, fs, fd, CVT_L_S);
3338 void Assembler::cvt_l_d(FPURegister fd, FPURegister fs) {
3339 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
3340 GenInstrRegister(COP1, D, f0, fs, fd, CVT_L_D);
3344 void Assembler::trunc_l_s(FPURegister fd, FPURegister fs) {
3345 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
3346 GenInstrRegister(COP1, S, f0, fs, fd, TRUNC_L_S);
3350 void Assembler::trunc_l_d(FPURegister fd, FPURegister fs) {
3351 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
3352 GenInstrRegister(COP1, D, f0, fs, fd, TRUNC_L_D);
3356 void Assembler::round_l_s(FPURegister fd, FPURegister fs) {
3357 GenInstrRegister(COP1, S, f0, fs, fd, ROUND_L_S);
3361 void Assembler::round_l_d(FPURegister fd, FPURegister fs) {
3362 GenInstrRegister(COP1, D, f0, fs, fd, ROUND_L_D);
3366 void Assembler::floor_l_s(FPURegister fd, FPURegister fs) {
3367 GenInstrRegister(COP1, S, f0, fs, fd, FLOOR_L_S);
3371 void Assembler::floor_l_d(FPURegister fd, FPURegister fs) {
3372 GenInstrRegister(COP1, D, f0, fs, fd, FLOOR_L_D);
3376 void Assembler::ceil_l_s(FPURegister fd, FPURegister fs) {
3377 GenInstrRegister(COP1, S, f0, fs, fd, CEIL_L_S);
3381 void Assembler::ceil_l_d(FPURegister fd, FPURegister fs) {
3382 GenInstrRegister(COP1, D, f0, fs, fd, CEIL_L_D);
3386 void Assembler::class_s(FPURegister fd, FPURegister fs) {
3387 DCHECK_EQ(kArchVariant, kMips64r6);
3388 GenInstrRegister(COP1, S, f0, fs, fd, CLASS_S);
3392 void Assembler::class_d(FPURegister fd, FPURegister fs) {
3393 DCHECK_EQ(kArchVariant, kMips64r6);
3394 GenInstrRegister(COP1, D, f0, fs, fd, CLASS_D);
3398 void Assembler::mina(SecondaryField fmt, FPURegister fd, FPURegister fs,
3400 DCHECK_EQ(kArchVariant, kMips64r6);
3401 DCHECK((fmt == D) || (fmt == S));
3402 GenInstrRegister(COP1, fmt, ft, fs, fd, MINA);
3406 void Assembler::maxa(SecondaryField fmt, FPURegister fd, FPURegister fs,
3408 DCHECK_EQ(kArchVariant, kMips64r6);
3409 DCHECK((fmt == D) || (fmt == S));
3410 GenInstrRegister(COP1, fmt, ft, fs, fd, MAXA);
3414 void Assembler::cvt_s_w(FPURegister fd, FPURegister fs) {
3415 GenInstrRegister(COP1, W, f0, fs, fd, CVT_S_W);
3419 void Assembler::cvt_s_l(FPURegister fd, FPURegister fs) {
3420 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
3421 GenInstrRegister(COP1, L, f0, fs, fd, CVT_S_L);
3425 void Assembler::cvt_s_d(FPURegister fd, FPURegister fs) {
3426 GenInstrRegister(COP1, D, f0, fs, fd, CVT_S_D);
3430 void Assembler::cvt_d_w(FPURegister fd, FPURegister fs) {
3431 GenInstrRegister(COP1, W, f0, fs, fd, CVT_D_W);
3435 void Assembler::cvt_d_l(FPURegister fd, FPURegister fs) {
3436 DCHECK(kArchVariant == kMips64r2 || kArchVariant == kMips64r6);
3437 GenInstrRegister(COP1, L, f0, fs, fd, CVT_D_L);
3441 void Assembler::cvt_d_s(FPURegister fd, FPURegister fs) {
3442 GenInstrRegister(COP1, S, f0, fs, fd, CVT_D_S);
3447 void Assembler::cmp(FPUCondition cond, SecondaryField fmt,
3448 FPURegister fd, FPURegister fs, FPURegister ft) {
3449 DCHECK_EQ(kArchVariant, kMips64r6);
3450 DCHECK_EQ(fmt & ~(31 << kRsShift), 0);
3451 Instr instr = COP1 | fmt | ft.code() << kFtShift |
3452 fs.code() << kFsShift | fd.code() << kFdShift | (0 << 5) | cond;
3457 void Assembler::cmp_s(FPUCondition cond, FPURegister fd, FPURegister fs,
3459 cmp(cond, W, fd, fs, ft);
3462 void Assembler::cmp_d(FPUCondition cond, FPURegister fd, FPURegister fs,
3464 cmp(cond, L, fd, fs, ft);
3468 void Assembler::bc1eqz(int16_t offset, FPURegister ft) {
3469 DCHECK_EQ(kArchVariant, kMips64r6);
3470 BlockTrampolinePoolScope block_trampoline_pool(
this);
3471 Instr instr = COP1 | BC1EQZ | ft.code() << kFtShift | (offset & kImm16Mask);
3473 BlockTrampolinePoolFor(1);
3477 void Assembler::bc1nez(int16_t offset, FPURegister ft) {
3478 DCHECK_EQ(kArchVariant, kMips64r6);
3479 BlockTrampolinePoolScope block_trampoline_pool(
this);
3480 Instr instr = COP1 | BC1NEZ | ft.code() << kFtShift | (offset & kImm16Mask);
3482 BlockTrampolinePoolFor(1);
3487 void Assembler::c(FPUCondition cond, SecondaryField fmt,
3488 FPURegister fs, FPURegister ft, uint16_t cc) {
3489 DCHECK_NE(kArchVariant, kMips64r6);
3490 DCHECK(is_uint3(cc));
3491 DCHECK(fmt == S || fmt == D);
3492 DCHECK_EQ(fmt & ~(31 << kRsShift), 0);
3493 Instr instr = COP1 | fmt | ft.code() << kFtShift | fs.code() << kFsShift
3494 | cc << 8 | 3 << 4 | cond;
3499 void Assembler::c_s(FPUCondition cond, FPURegister fs, FPURegister ft,
3501 c(cond, S, fs, ft, cc);
3505 void Assembler::c_d(FPUCondition cond, FPURegister fs, FPURegister ft,
3507 c(cond, D, fs, ft, cc);
3511 void Assembler::fcmp(FPURegister src1,
const double src2,
3512 FPUCondition cond) {
3513 DCHECK_EQ(src2, 0.0);
3514 mtc1(zero_reg, f14);
3516 c(cond, D, src1, f14, 0);
3520 void Assembler::bc1f(int16_t offset, uint16_t cc) {
3521 BlockTrampolinePoolScope block_trampoline_pool(
this);
3522 DCHECK(is_uint3(cc));
3523 Instr instr = COP1 | BC1 | cc << 18 | 0 << 16 | (offset & kImm16Mask);
3525 BlockTrampolinePoolFor(1);
3529 void Assembler::bc1t(int16_t offset, uint16_t cc) {
3530 BlockTrampolinePoolScope block_trampoline_pool(
this);
3531 DCHECK(is_uint3(cc));
3532 Instr instr = COP1 | BC1 | cc << 18 | 1 << 16 | (offset & kImm16Mask);
3534 BlockTrampolinePoolFor(1);
3538 #define MSA_BRANCH_LIST(V) \ 3550 #define MSA_BRANCH(name, opcode) \ 3551 void Assembler::name(MSARegister wt, int16_t offset) { \ 3552 GenInstrMsaBranch(opcode, wt, offset); \ 3555 MSA_BRANCH_LIST(MSA_BRANCH)
3557 #undef MSA_BRANCH_LIST 3559 #define MSA_LD_ST_LIST(V) \ 3569 #define MSA_LD_ST(name, opcode) \ 3570 void Assembler::name(MSARegister wd, const MemOperand& rs) { \ 3571 MemOperand source = rs; \ 3572 AdjustBaseAndOffset(source); \ 3573 if (is_int10(source.offset())) { \ 3574 GenInstrMsaMI10(opcode, source.offset(), source.rm(), wd); \ 3576 UseScratchRegisterScope temps(this); \ 3577 Register scratch = temps.Acquire(); \ 3578 DCHECK(rs.rm() != scratch); \ 3579 daddiu(scratch, source.rm(), source.offset()); \ 3580 GenInstrMsaMI10(opcode, 0, scratch, wd); \ 3584 MSA_LD_ST_LIST(MSA_LD_ST)
3586 #undef MSA_BRANCH_LIST 3588 #define MSA_I10_LIST(V) \ 3594 #define MSA_I10(name, format) \ 3595 void Assembler::name(MSARegister wd, int32_t imm10) { \ 3596 GenInstrMsaI10(LDI, format, imm10, wd); \ 3598 MSA_I10_LIST(MSA_I10)
3602 #define MSA_I5_LIST(V) \ 3615 #define MSA_I5_FORMAT(name, opcode, format) \ 3616 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 3618 GenInstrMsaI5(opcode, I5_DF_##format, imm5, ws, wd); \ 3621 #define MSA_I5(name, opcode) \ 3622 MSA_I5_FORMAT(name, opcode, b) \ 3623 MSA_I5_FORMAT(name, opcode, h) \ 3624 MSA_I5_FORMAT(name, opcode, w) \ 3625 MSA_I5_FORMAT(name, opcode, d) 3629 #undef MSA_I5_FORMAT 3632 #define MSA_I8_LIST(V) \ 3637 V(bmnzi_b, BMNZI_B) \ 3639 V(bseli_b, BSELI_B) \ 3644 #define MSA_I8(name, opcode) \ 3645 void Assembler::name(MSARegister wd, MSARegister ws, uint32_t imm8) { \ 3646 GenInstrMsaI8(opcode, imm8, ws, wd); \ 3653 #define MSA_VEC_LIST(V) \ 3662 #define MSA_VEC(name, opcode) \ 3663 void Assembler::name(MSARegister wd, MSARegister ws, MSARegister wt) { \ 3664 GenInstrMsaVec(opcode, wt, ws, wd); \ 3667 MSA_VEC_LIST(MSA_VEC)
3671 #define MSA_2R_LIST(V) \ 3676 #define MSA_2R_FORMAT(name, opcode, format) \ 3677 void Assembler::name##_##format(MSARegister wd, MSARegister ws) { \ 3678 GenInstrMsa2R(opcode, MSA_2R_DF_##format, ws, wd); \ 3681 #define MSA_2R(name, opcode) \ 3682 MSA_2R_FORMAT(name, opcode, b) \ 3683 MSA_2R_FORMAT(name, opcode, h) \ 3684 MSA_2R_FORMAT(name, opcode, w) \ 3685 MSA_2R_FORMAT(name, opcode, d) 3689 #undef MSA_2R_FORMAT 3692 #define MSA_FILL(format) \ 3693 void Assembler::fill_##format(MSARegister wd, Register rs) { \ 3694 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD)); \ 3695 DCHECK(rs.is_valid() && wd.is_valid()); \ 3696 Instr instr = MSA | MSA_2R_FORMAT | FILL | MSA_2R_DF_##format | \ 3697 (rs.code() << kWsShift) | (wd.code() << kWdShift) | \ 3698 MSA_VEC_2R_2RF_MINOR; \ 3708 #define MSA_2RF_LIST(V) \ 3710 V(ftrunc_s, FTRUNC_S) \ 3711 V(ftrunc_u, FTRUNC_U) \ 3721 V(ftint_s, FTINT_S) \ 3722 V(ftint_u, FTINT_U) \ 3723 V(ffint_s, FFINT_S) \ 3726 #define MSA_2RF_FORMAT(name, opcode, format) \ 3727 void Assembler::name##_##format(MSARegister wd, MSARegister ws) { \ 3728 GenInstrMsa2RF(opcode, MSA_2RF_DF_##format, ws, wd); \ 3731 #define MSA_2RF(name, opcode) \ 3732 MSA_2RF_FORMAT(name, opcode, w) \ 3733 MSA_2RF_FORMAT(name, opcode, d) 3735 MSA_2RF_LIST(MSA_2RF)
3737 #undef MSA_2RF_FORMAT 3740 #define MSA_3R_LIST(V) \ 3772 V(subsus_u, SUBSUS_U) \ 3773 V(subsuu_s, SUBSUU_S) \ 3779 V(div_s, DIV_S_MSA) \ 3785 V(dpadd_s, DPADD_S) \ 3786 V(dpadd_u, DPADD_U) \ 3787 V(dpsub_s, DPSUB_S) \ 3788 V(dpsub_u, DPSUB_U) \ 3803 #define MSA_3R_FORMAT(name, opcode, format) \ 3804 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 3806 GenInstrMsa3R<MSARegister>(opcode, MSA_3R_DF_##format, wt, ws, wd); \ 3809 #define MSA_3R_FORMAT_SLD_SPLAT(name, opcode, format) \ 3810 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 3812 GenInstrMsa3R<Register>(opcode, MSA_3R_DF_##format, rt, ws, wd); \ 3815 #define MSA_3R(name, opcode) \ 3816 MSA_3R_FORMAT(name, opcode, b) \ 3817 MSA_3R_FORMAT(name, opcode, h) \ 3818 MSA_3R_FORMAT(name, opcode, w) \ 3819 MSA_3R_FORMAT(name, opcode, d) 3821 #define MSA_3R_SLD_SPLAT(name, opcode) \ 3822 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, b) \ 3823 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, h) \ 3824 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, w) \ 3825 MSA_3R_FORMAT_SLD_SPLAT(name, opcode, d) 3828 MSA_3R_SLD_SPLAT(sld, SLD)
3829 MSA_3R_SLD_SPLAT(splat, SPLAT)
3832 #undef MSA_3R_FORMAT 3833 #undef MSA_3R_FORMAT_SLD_SPLAT 3834 #undef MSA_3R_SLD_SPLAT 3837 #define MSA_3RF_LIST1(V) \ 3872 #define MSA_3RF_LIST2(V) \ 3879 V(maddr_q, MADDR_Q) \ 3882 #define MSA_3RF_FORMAT(name, opcode, df, df_c) \ 3883 void Assembler::name##_##df(MSARegister wd, MSARegister ws, \ 3885 GenInstrMsa3RF(opcode, df_c, wt, ws, wd); \ 3888 #define MSA_3RF_1(name, opcode) \ 3889 MSA_3RF_FORMAT(name, opcode, w, 0) \ 3890 MSA_3RF_FORMAT(name, opcode, d, 1) 3892 #define MSA_3RF_2(name, opcode) \ 3893 MSA_3RF_FORMAT(name, opcode, h, 0) \ 3894 MSA_3RF_FORMAT(name, opcode, w, 1) 3896 MSA_3RF_LIST1(MSA_3RF_1)
3897 MSA_3RF_LIST2(MSA_3RF_2)
3900 #undef MSA_3RF_FORMAT 3901 #undef MSA_3RF_LIST1 3902 #undef MSA_3RF_LIST2 3904 void Assembler::sldi_b(MSARegister wd, MSARegister ws,
uint32_t n) {
3905 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_B, n, ws, wd);
3908 void Assembler::sldi_h(MSARegister wd, MSARegister ws,
uint32_t n) {
3909 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_H, n, ws, wd);
3912 void Assembler::sldi_w(MSARegister wd, MSARegister ws,
uint32_t n) {
3913 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_W, n, ws, wd);
3916 void Assembler::sldi_d(MSARegister wd, MSARegister ws,
uint32_t n) {
3917 GenInstrMsaElm<MSARegister, MSARegister>(SLDI, ELM_DF_D, n, ws, wd);
3920 void Assembler::splati_b(MSARegister wd, MSARegister ws,
uint32_t n) {
3921 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_B, n, ws, wd);
3924 void Assembler::splati_h(MSARegister wd, MSARegister ws,
uint32_t n) {
3925 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_H, n, ws, wd);
3928 void Assembler::splati_w(MSARegister wd, MSARegister ws,
uint32_t n) {
3929 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_W, n, ws, wd);
3932 void Assembler::splati_d(MSARegister wd, MSARegister ws,
uint32_t n) {
3933 GenInstrMsaElm<MSARegister, MSARegister>(SPLATI, ELM_DF_D, n, ws, wd);
3936 void Assembler::copy_s_b(Register rd, MSARegister ws,
uint32_t n) {
3937 GenInstrMsaElm<Register, MSARegister>(COPY_S, ELM_DF_B, n, ws, rd);
3940 void Assembler::copy_s_h(Register rd, MSARegister ws,
uint32_t n) {
3941 GenInstrMsaElm<Register, MSARegister>(COPY_S, ELM_DF_H, n, ws, rd);
3944 void Assembler::copy_s_w(Register rd, MSARegister ws,
uint32_t n) {
3945 GenInstrMsaElm<Register, MSARegister>(COPY_S, ELM_DF_W, n, ws, rd);
3948 void Assembler::copy_s_d(Register rd, MSARegister ws,
uint32_t n) {
3949 GenInstrMsaElm<Register, MSARegister>(COPY_S, ELM_DF_D, n, ws, rd);
3952 void Assembler::copy_u_b(Register rd, MSARegister ws,
uint32_t n) {
3953 GenInstrMsaElm<Register, MSARegister>(COPY_U, ELM_DF_B, n, ws, rd);
3956 void Assembler::copy_u_h(Register rd, MSARegister ws,
uint32_t n) {
3957 GenInstrMsaElm<Register, MSARegister>(COPY_U, ELM_DF_H, n, ws, rd);
3960 void Assembler::copy_u_w(Register rd, MSARegister ws,
uint32_t n) {
3961 GenInstrMsaElm<Register, MSARegister>(COPY_U, ELM_DF_W, n, ws, rd);
3964 void Assembler::insert_b(MSARegister wd,
uint32_t n, Register rs) {
3965 GenInstrMsaElm<MSARegister, Register>(INSERT, ELM_DF_B, n, rs, wd);
3968 void Assembler::insert_h(MSARegister wd,
uint32_t n, Register rs) {
3969 GenInstrMsaElm<MSARegister, Register>(INSERT, ELM_DF_H, n, rs, wd);
3972 void Assembler::insert_w(MSARegister wd,
uint32_t n, Register rs) {
3973 GenInstrMsaElm<MSARegister, Register>(INSERT, ELM_DF_W, n, rs, wd);
3976 void Assembler::insert_d(MSARegister wd,
uint32_t n, Register rs) {
3977 GenInstrMsaElm<MSARegister, Register>(INSERT, ELM_DF_D, n, rs, wd);
3980 void Assembler::insve_b(MSARegister wd,
uint32_t n, MSARegister ws) {
3981 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_B, n, ws, wd);
3984 void Assembler::insve_h(MSARegister wd,
uint32_t n, MSARegister ws) {
3985 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_H, n, ws, wd);
3988 void Assembler::insve_w(MSARegister wd,
uint32_t n, MSARegister ws) {
3989 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_W, n, ws, wd);
3992 void Assembler::insve_d(MSARegister wd,
uint32_t n, MSARegister ws) {
3993 GenInstrMsaElm<MSARegister, MSARegister>(INSVE, ELM_DF_D, n, ws, wd);
3996 void Assembler::move_v(MSARegister wd, MSARegister ws) {
3997 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
3998 DCHECK(ws.is_valid() && wd.is_valid());
3999 Instr instr = MSA | MOVE_V | (ws.code() << kWsShift) |
4000 (wd.code() << kWdShift) | MSA_ELM_MINOR;
4004 void Assembler::ctcmsa(MSAControlRegister cd, Register rs) {
4005 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
4006 DCHECK(cd.is_valid() && rs.is_valid());
4007 Instr instr = MSA | CTCMSA | (rs.code() << kWsShift) |
4008 (cd.code() << kWdShift) | MSA_ELM_MINOR;
4012 void Assembler::cfcmsa(Register rd, MSAControlRegister cs) {
4013 DCHECK((kArchVariant == kMips64r6) && IsEnabled(MIPS_SIMD));
4014 DCHECK(rd.is_valid() && cs.is_valid());
4015 Instr instr = MSA | CFCMSA | (cs.code() << kWsShift) |
4016 (rd.code() << kWdShift) | MSA_ELM_MINOR;
4020 #define MSA_BIT_LIST(V) \ 4034 #define MSA_BIT_FORMAT(name, opcode, format) \ 4035 void Assembler::name##_##format(MSARegister wd, MSARegister ws, \ 4037 GenInstrMsaBit(opcode, BIT_DF_##format, m, ws, wd); \ 4040 #define MSA_BIT(name, opcode) \ 4041 MSA_BIT_FORMAT(name, opcode, b) \ 4042 MSA_BIT_FORMAT(name, opcode, h) \ 4043 MSA_BIT_FORMAT(name, opcode, w) \ 4044 MSA_BIT_FORMAT(name, opcode, d) 4046 MSA_BIT_LIST(MSA_BIT)
4048 #undef MSA_BIT_FORMAT 4051 int Assembler::RelocateInternalReference(RelocInfo::Mode rmode, Address pc,
4052 intptr_t pc_delta) {
4053 if (RelocInfo::IsInternalReference(rmode)) {
4055 if (*p == kEndOfJumpChain) {
4061 Instr instr = instr_at(pc);
4062 DCHECK(RelocInfo::IsInternalReferenceEncoded(rmode));
4064 Instr instr_lui = instr_at(pc + 0 * kInstrSize);
4065 Instr instr_ori = instr_at(pc + 1 * kInstrSize);
4066 Instr instr_ori2 = instr_at(pc + 3 * kInstrSize);
4067 DCHECK(IsOri(instr_ori));
4068 DCHECK(IsOri(instr_ori2));
4070 int64_t imm = (instr_lui &
static_cast<int64_t>(kImm16Mask)) << 48;
4071 imm |= (instr_ori &
static_cast<int64_t>(kImm16Mask)) << 32;
4072 imm |= (instr_ori2 &
static_cast<int64_t>(kImm16Mask)) << 16;
4076 if (imm == kEndOfJumpChain) {
4080 DCHECK_EQ(imm & 3, 0);
4082 instr_lui &= ~kImm16Mask;
4083 instr_ori &= ~kImm16Mask;
4084 instr_ori2 &= ~kImm16Mask;
4086 instr_at_put(pc + 0 * kInstrSize, instr_lui | ((imm >> 32) & kImm16Mask));
4087 instr_at_put(pc + 1 * kInstrSize, instr_ori | (imm >> 16 & kImm16Mask));
4088 instr_at_put(pc + 3 * kInstrSize, instr_ori2 | (imm & kImm16Mask));
4090 }
else if (IsJ(instr) || IsJal(instr)) {
4092 uint32_t imm28 = (instr &
static_cast<int32_t
>(kImm26Mask)) << 2;
4094 imm28 &= kImm28Mask;
4095 instr &= ~kImm26Mask;
4096 DCHECK_EQ(imm28 & 3, 0);
4098 instr_at_put(pc, instr | (imm26 & kImm26Mask));
4101 DCHECK(((instr & kJumpRawMask) == kJRawMark) ||
4102 ((instr & kJumpRawMask) == kJalRawMark));
4104 int32_t imm28 = (instr &
static_cast<int32_t
>(kImm26Mask)) << 2;
4106 imm28 = (imm28 << 4) >> 4;
4108 static_cast<int64_t>(imm28) + reinterpret_cast<uint64_t>(pc);
4109 target &= kImm28Mask;
4110 DCHECK_EQ(imm28 & 3, 0);
4113 uint32_t unbox = (instr & kJRawMark) ? J : JAL;
4114 instr_at_put(pc, unbox | (imm26 & kImm26Mask));
4120 void Assembler::GrowBuffer() {
4121 if (!own_buffer_) FATAL(
"external code buffer is too small");
4125 if (buffer_size_ < 1 * MB) {
4126 desc.buffer_size = 2*buffer_size_;
4128 desc.buffer_size = buffer_size_ + 1*MB;
4133 if (desc.buffer_size > kMaximalBufferSize) {
4134 V8::FatalProcessOutOfMemory(
nullptr,
"Assembler::GrowBuffer");
4138 desc.buffer = NewArray<byte>(desc.buffer_size);
4141 desc.instr_size = pc_offset();
4143 static_cast<int>((buffer_ + buffer_size_) - reloc_info_writer.pos());
4146 intptr_t pc_delta = desc.buffer - buffer_;
4147 intptr_t rc_delta = (desc.buffer + desc.buffer_size) -
4148 (buffer_ + buffer_size_);
4149 MemMove(desc.buffer, buffer_, desc.instr_size);
4150 MemMove(reloc_info_writer.pos() + rc_delta,
4151 reloc_info_writer.pos(), desc.reloc_size);
4154 DeleteArray(buffer_);
4155 buffer_ = desc.buffer;
4156 buffer_size_ = desc.buffer_size;
4158 reloc_info_writer.Reposition(reloc_info_writer.pos() + rc_delta,
4159 reloc_info_writer.last_pc() + pc_delta);
4162 for (RelocIterator it(desc); !it.done(); it.next()) {
4163 RelocInfo::Mode rmode = it.rinfo()->rmode();
4164 if (rmode == RelocInfo::INTERNAL_REFERENCE) {
4165 RelocateInternalReference(rmode, it.rinfo()->pc(), pc_delta);
4168 DCHECK(!overflow());
4172 void Assembler::db(uint8_t data) {
4173 CheckForEmitInForbiddenSlot();
4178 void Assembler::dd(
uint32_t data) {
4179 CheckForEmitInForbiddenSlot();
4184 void Assembler::dq(uint64_t data) {
4185 CheckForEmitInForbiddenSlot();
4190 void Assembler::dd(Label* label) {
4192 CheckForEmitInForbiddenSlot();
4193 if (label->is_bound()) {
4194 data =
reinterpret_cast<uint64_t
>(buffer_ + label->pos());
4196 data = jump_address(label);
4197 unbound_labels_count_++;
4198 internal_reference_positions_.insert(label->pos());
4200 RecordRelocInfo(RelocInfo::INTERNAL_REFERENCE);
4205 void Assembler::RecordRelocInfo(RelocInfo::Mode rmode, intptr_t data) {
4206 if (!ShouldRecordRelocInfo(rmode))
return;
4208 RelocInfo rinfo(reinterpret_cast<Address>(pc_), rmode, data, Code());
4209 DCHECK_GE(buffer_space(), kMaxRelocSize);
4210 reloc_info_writer.Write(&rinfo);
4214 void Assembler::BlockTrampolinePoolFor(
int instructions) {
4215 CheckTrampolinePoolQuick(instructions);
4216 BlockTrampolinePoolBefore(pc_offset() + instructions * kInstrSize);
4220 void Assembler::CheckTrampolinePool() {
4226 if ((trampoline_pool_blocked_nesting_ > 0) ||
4227 (pc_offset() < no_trampoline_pool_before_)) {
4230 if (trampoline_pool_blocked_nesting_ > 0) {
4231 next_buffer_check_ = pc_offset() + kInstrSize;
4233 next_buffer_check_ = no_trampoline_pool_before_;
4238 DCHECK(!trampoline_emitted_);
4239 DCHECK_GE(unbound_labels_count_, 0);
4240 if (unbound_labels_count_ > 0) {
4242 { BlockTrampolinePoolScope block_trampoline_pool(
this);
4244 if (kArchVariant == kMips64r6) {
4251 int pool_start = pc_offset();
4252 for (
int i = 0;
i < unbound_labels_count_;
i++) {
4254 if (kArchVariant == kMips64r6) {
4258 or_(t8, ra, zero_reg);
4263 or_(ra, t8, zero_reg);
4273 trampoline_ = Trampoline(pool_start, unbound_labels_count_);
4275 trampoline_emitted_ =
true;
4278 next_buffer_check_ = kMaxInt;
4283 next_buffer_check_ = pc_offset() +
4284 kMaxBranchOffset - kTrampolineSlotsSize * 16;
4290 Address Assembler::target_address_at(Address pc) {
4291 Instr instr0 = instr_at(pc);
4292 Instr instr1 = instr_at(pc + 1 * kInstrSize);
4293 Instr instr3 = instr_at(pc + 3 * kInstrSize);
4297 if ((GetOpcodeField(instr0) == LUI) && (GetOpcodeField(instr1) == ORI) &&
4298 (GetOpcodeField(instr3) == ORI)) {
4301 ((uint64_t)(GetImmediate16(instr0)) << 32) |
4302 ((uint64_t)(GetImmediate16(instr1)) << 16) |
4303 ((uint64_t)(GetImmediate16(instr3))));
4306 addr = (addr << 16) >> 16;
4307 return static_cast<Address
>(addr);
4318 void Assembler::QuietNaN(HeapObject*
object) {
4319 HeapNumber::cast(
object)->set_value(std::numeric_limits<double>::quiet_NaN());
4335 void Assembler::set_target_value_at(Address pc, uint64_t target,
4336 ICacheFlushMode icache_flush_mode) {
4342 Instr instr1 = instr_at(pc + kInstrSize);
4348 Instr instr0 = instr_at(pc);
4349 Instr instr3 = instr_at(pc + kInstrSize * 3);
4350 DCHECK((GetOpcodeField(instr0) == LUI && GetOpcodeField(instr1) == ORI &&
4351 GetOpcodeField(instr3) == ORI));
4359 *p = LUI | (rt_code << kRtShift) | ((target >> 32) & kImm16Mask);
4360 *(p + 1) = ORI | (rt_code << kRtShift) | (rt_code << kRsShift) |
4361 ((target >> 16) & kImm16Mask);
4362 *(p + 3) = ORI | (rt_code << kRsShift) | (rt_code << kRtShift) |
4363 (target & kImm16Mask);
4365 if (icache_flush_mode != SKIP_ICACHE_FLUSH) {
4366 Assembler::FlushICache(pc, 4 * kInstrSize);
4370 UseScratchRegisterScope::UseScratchRegisterScope(Assembler* assembler)
4371 : available_(assembler->GetScratchRegisterList()),
4372 old_available_(*available_) {}
4374 UseScratchRegisterScope::~UseScratchRegisterScope() {
4375 *available_ = old_available_;
4378 Register UseScratchRegisterScope::Acquire() {
4379 DCHECK_NOT_NULL(available_);
4380 DCHECK_NE(*available_, 0);
4381 int index =
static_cast<int>(base::bits::CountTrailingZeros32(*available_));
4382 *available_ &= ~(1UL << index);
4384 return Register::from_code(index);
4387 bool UseScratchRegisterScope::hasAvailable()
const {
return *available_ != 0; }
4392 #endif // V8_TARGET_ARCH_MIPS64