/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: set ts=8 sts=2 et sw=2 tw=80: * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ // Copyright 2021 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "jit/riscv64/MacroAssembler-riscv64.h" #include "jsmath.h" #include "jit/Bailouts.h" #include "jit/BaselineFrame.h" #include "jit/JitFrames.h" #include "jit/JitRuntime.h" #include "jit/MacroAssembler.h" #include "jit/MoveEmitter.h" #include "jit/riscv64/SharedICRegisters-riscv64.h" #include "util/Memory.h" #include "vm/JitActivation.h" // jit::JitActivation #include "vm/JSContext.h" #include "jit/MacroAssembler-inl.h" namespace js { namespace jit { MacroAssembler& MacroAssemblerRiscv64::asMasm() { return *static_cast(this); } const MacroAssembler& MacroAssemblerRiscv64::asMasm() const { return *static_cast(this); } void MacroAssemblerRiscv64::ma_cmp_set(Register rd, Register rj, ImmWord imm, Condition c) { if (imm.value <= INT32_MAX) { ma_cmp_set(rd, rj, Imm32(uint32_t(imm.value)), c); } else { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, imm); ma_cmp_set(rd, rj, scratch, c); } } void MacroAssemblerRiscv64::ma_cmp_set(Register rd, Register rj, ImmPtr imm, Condition c) { ma_cmp_set(rd, rj, ImmWord(uintptr_t(imm.value)), c); } void MacroAssemblerRiscv64::ma_cmp_set(Register rd, Address address, Imm32 imm, Condition c) { // TODO(riscv): 32-bit ma_cmp_set? UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); ma_load(scratch2, address, SizeWord); ma_cmp_set(rd, Register(scratch2), imm, c); } void MacroAssemblerRiscv64::ma_cmp_set(Register rd, Address address, ImmWord imm, Condition c) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); ma_load(scratch2, address, SizeDouble); ma_cmp_set(rd, Register(scratch2), imm, c); } void MacroAssemblerRiscv64::ma_cmp_set(Register rd, Register rj, Imm32 imm, Condition c) { if (imm.value == 0) { switch (c) { case Equal: case BelowOrEqual: ma_sltu(rd, rj, Operand(1)); break; case NotEqual: case Above: sltu(rd, zero, rj); break; case AboveOrEqual: case Below: ori(rd, zero, c == AboveOrEqual ? 1 : 0); break; case GreaterThan: case LessThanOrEqual: slt(rd, zero, rj); if (c == LessThanOrEqual) { xori(rd, rd, 1); } break; case LessThan: case GreaterThanOrEqual: slt(rd, rj, zero); if (c == GreaterThanOrEqual) { xori(rd, rd, 1); } break; case Zero: ma_sltu(rd, rj, Operand(1)); break; case NonZero: sltu(rd, zero, rj); break; case Signed: slt(rd, rj, zero); break; case NotSigned: slt(rd, rj, zero); xori(rd, rd, 1); break; default: MOZ_CRASH("Invalid condition."); } return; } switch (c) { case Equal: case NotEqual: ma_xor(rd, rj, imm); if (c == Equal) { ma_sltu(rd, rd, Operand(1)); } else { sltu(rd, zero, rd); } break; case Zero: case NonZero: case Signed: case NotSigned: MOZ_CRASH("Invalid condition."); default: Condition cond = ma_cmp(rd, rj, imm, c); MOZ_ASSERT(cond == Equal || cond == NotEqual); if (cond == Equal) xori(rd, rd, 1); } } Assembler::Condition MacroAssemblerRiscv64::ma_cmp(Register dest, Register lhs, Register rhs, Condition c) { switch (c) { case Above: // bgtu s,t,label => // sltu at,t,s // bne at,$zero,offs sltu(dest, rhs, lhs); return NotEqual; case AboveOrEqual: // bgeu s,t,label => // sltu at,s,t // beq at,$zero,offs sltu(dest, lhs, rhs); return Equal; case Below: // bltu s,t,label => // sltu at,s,t // bne at,$zero,offs sltu(dest, lhs, rhs); return NotEqual; case BelowOrEqual: // bleu s,t,label => // sltu at,t,s // beq at,$zero,offs sltu(dest, rhs, lhs); return Equal; case GreaterThan: // bgt s,t,label => // slt at,t,s // bne at,$zero,offs slt(dest, rhs, lhs); return NotEqual; case GreaterThanOrEqual: // bge s,t,label => // slt at,s,t // beq at,$zero,offs slt(dest, lhs, rhs); return Equal; case LessThan: // blt s,t,label => // slt at,s,t // bne at,$zero,offs slt(dest, lhs, rhs); return NotEqual; case LessThanOrEqual: // ble s,t,label => // slt at,t,s // beq at,$zero,offs slt(dest, rhs, lhs); return Equal; default: MOZ_CRASH("Invalid condition."); } return Always; } Assembler::Condition MacroAssemblerRiscv64::ma_cmp(Register dest, Register lhs, Imm32 imm, Condition c) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_RELEASE_ASSERT(lhs != scratch); switch (c) { case Above: case BelowOrEqual: if (imm.value != 0x7fffffff && is_intn(imm.value + 1, 12) && imm.value != -1) { // lhs <= rhs via lhs < rhs + 1 if rhs + 1 does not overflow ma_sltu(dest, lhs, Operand(imm.value + 1)); return (c == BelowOrEqual ? NotEqual : Equal); } else { ma_li(scratch, imm); sltu(dest, scratch, lhs); return (c == BelowOrEqual ? Equal : NotEqual); } case AboveOrEqual: case Below: if (is_intn(imm.value, 12)) { ma_sltu(dest, lhs, Operand(imm.value)); } else { ma_li(scratch, imm); sltu(dest, lhs, scratch); } return (c == AboveOrEqual ? Equal : NotEqual); case GreaterThan: case LessThanOrEqual: if (imm.value != 0x7fffffff && is_intn(imm.value + 1, 12)) { // lhs <= rhs via lhs < rhs + 1. ma_slt(dest, lhs, Operand(imm.value + 1)); return (c == LessThanOrEqual ? NotEqual : Equal); } else { ma_li(scratch, imm); slt(dest, scratch, lhs); return (c == LessThanOrEqual ? Equal : NotEqual); } case GreaterThanOrEqual: case LessThan: if (is_intn(imm.value, 12)) { ma_slt(dest, lhs, imm); } else { ma_li(scratch, imm); slt(dest, lhs, scratch); } return (c == GreaterThanOrEqual ? Equal : NotEqual); default: MOZ_CRASH("Invalid condition."); } return Always; } void MacroAssemblerRiscv64::ma_cmp_set(Register rd, Register rj, Register rk, Condition c) { switch (c) { case Equal: // seq d,s,t => // xor d,s,t // sltiu d,d,1 xor_(rd, rj, rk); ma_sltu(rd, rd, Operand(1)); break; case NotEqual: // sne d,s,t => // xor d,s,t // sltu d,$zero,d xor_(rd, rj, rk); sltu(rd, zero, rd); break; case Above: // sgtu d,s,t => // sltu d,t,s sltu(rd, rk, rj); break; case AboveOrEqual: // sgeu d,s,t => // sltu d,s,t // xori d,d,1 sltu(rd, rj, rk); xori(rd, rd, 1); break; case Below: // sltu d,s,t sltu(rd, rj, rk); break; case BelowOrEqual: // sleu d,s,t => // sltu d,t,s // xori d,d,1 sltu(rd, rk, rj); xori(rd, rd, 1); break; case GreaterThan: // sgt d,s,t => // slt d,t,s slt(rd, rk, rj); break; case GreaterThanOrEqual: // sge d,s,t => // slt d,s,t // xori d,d,1 slt(rd, rj, rk); xori(rd, rd, 1); break; case LessThan: // slt d,s,t slt(rd, rj, rk); break; case LessThanOrEqual: // sle d,s,t => // slt d,t,s // xori d,d,1 slt(rd, rk, rj); xori(rd, rd, 1); break; case Zero: MOZ_ASSERT(rj == rk); // seq d,s,$zero => // sltiu d,s,1 ma_sltu(rd, rj, Operand(1)); break; case NonZero: MOZ_ASSERT(rj == rk); // sne d,s,$zero => // sltu d,$zero,s sltu(rd, zero, rj); break; case Signed: MOZ_ASSERT(rj == rk); slt(rd, rj, zero); break; case NotSigned: MOZ_ASSERT(rj == rk); // sge d,s,$zero => // slt d,s,$zero // xori d,d,1 slt(rd, rj, zero); xori(rd, rd, 1); break; default: MOZ_CRASH("Invalid condition."); } } void MacroAssemblerRiscv64::ma_compareF32(Register rd, DoubleCondition cc, FloatRegister cmp1, FloatRegister cmp2) { switch (cc) { case DoubleEqualOrUnordered: case DoubleEqual: feq_s(rd, cmp1, cmp2); break; case DoubleNotEqualOrUnordered: case DoubleNotEqual: { Label done; CompareIsNanF32(rd, cmp1, cmp2); ma_branch(&done, Equal, rd, Operand(1)); feq_s(rd, cmp1, cmp2); bind(&done); NegateBool(rd, rd); break; } case DoubleLessThanOrUnordered: case DoubleLessThan: flt_s(rd, cmp1, cmp2); break; case DoubleGreaterThanOrEqualOrUnordered: case DoubleGreaterThanOrEqual: fle_s(rd, cmp2, cmp1); break; case DoubleLessThanOrEqualOrUnordered: case DoubleLessThanOrEqual: fle_s(rd, cmp1, cmp2); break; case DoubleGreaterThanOrUnordered: case DoubleGreaterThan: flt_s(rd, cmp2, cmp1); break; case DoubleOrdered: CompareIsNotNanF32(rd, cmp1, cmp2); return; case DoubleUnordered: CompareIsNanF32(rd, cmp1, cmp2); return; } if (cc >= FIRST_UNORDERED && cc <= LAST_UNORDERED) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); CompareIsNanF32(scratch, cmp1, cmp2); or_(rd, rd, scratch); } } void MacroAssemblerRiscv64::ma_compareF64(Register rd, DoubleCondition cc, FloatRegister cmp1, FloatRegister cmp2) { switch (cc) { case DoubleEqualOrUnordered: case DoubleEqual: feq_d(rd, cmp1, cmp2); break; case DoubleNotEqualOrUnordered: case DoubleNotEqual: { Label done; CompareIsNanF64(rd, cmp1, cmp2); ma_branch(&done, Equal, rd, Operand(1)); feq_d(rd, cmp1, cmp2); bind(&done); NegateBool(rd, rd); } break; case DoubleLessThanOrUnordered: case DoubleLessThan: flt_d(rd, cmp1, cmp2); break; case DoubleGreaterThanOrEqualOrUnordered: case DoubleGreaterThanOrEqual: fle_d(rd, cmp2, cmp1); break; case DoubleLessThanOrEqualOrUnordered: case DoubleLessThanOrEqual: fle_d(rd, cmp1, cmp2); break; case DoubleGreaterThanOrUnordered: case DoubleGreaterThan: flt_d(rd, cmp2, cmp1); break; case DoubleOrdered: CompareIsNotNanF64(rd, cmp1, cmp2); return; case DoubleUnordered: CompareIsNanF64(rd, cmp1, cmp2); return; } if (cc >= FIRST_UNORDERED && cc <= LAST_UNORDERED) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); CompareIsNanF64(scratch, cmp1, cmp2); or_(rd, rd, scratch); } } void MacroAssemblerRiscv64Compat::movePtr(Register src, Register dest) { mv(dest, src); } void MacroAssemblerRiscv64Compat::movePtr(ImmWord imm, Register dest) { ma_li(dest, imm); } void MacroAssemblerRiscv64Compat::movePtr(ImmGCPtr imm, Register dest) { ma_li(dest, imm); } void MacroAssemblerRiscv64Compat::movePtr(ImmPtr imm, Register dest) { movePtr(ImmWord(uintptr_t(imm.value)), dest); } void MacroAssemblerRiscv64Compat::movePtr(wasm::SymbolicAddress imm, Register dest) { DEBUG_PRINTF("[ %s\n", __FUNCTION__); BlockTrampolinePoolScope block_trampoline_pool(this, 8); append(wasm::SymbolicAccess(CodeOffset(nextOffset().getOffset()), imm)); ma_liPatchable(dest, ImmWord(-1), Li64); DEBUG_PRINTF("]\n"); } bool MacroAssemblerRiscv64Compat::buildOOLFakeExitFrame(void* fakeReturnAddr) { asMasm().PushFrameDescriptor(FrameType::IonJS); // descriptor_ asMasm().Push(ImmPtr(fakeReturnAddr)); asMasm().Push(FramePointer); return true; } void MacroAssemblerRiscv64Compat::convertUInt32ToDouble(Register src, FloatRegister dest) { fcvt_d_wu(dest, src); } void MacroAssemblerRiscv64Compat::convertUInt64ToDouble(Register src, FloatRegister dest) { fcvt_d_lu(dest, src); } void MacroAssemblerRiscv64Compat::convertUInt32ToFloat32(Register src, FloatRegister dest) { fcvt_s_wu(dest, src); } void MacroAssemblerRiscv64Compat::convertDoubleToFloat32(FloatRegister src, FloatRegister dest) { fcvt_s_d(dest, src); } template void MacroAssemblerRiscv64::RoundHelper(FPURegister dst, FPURegister src, FPURegister fpu_scratch, FPURoundingMode frm) { BlockTrampolinePoolScope block_trampoline_pool(this, 20); UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); MOZ_ASSERT((std::is_same::value) || (std::is_same::value)); // Need at least two FPRs, so check against dst == src == fpu_scratch MOZ_ASSERT(!(dst == src && dst == fpu_scratch)); const int kFloatMantissaBits = sizeof(F) == 4 ? kFloat32MantissaBits : kFloat64MantissaBits; const int kFloatExponentBits = sizeof(F) == 4 ? kFloat32ExponentBits : kFloat64ExponentBits; const int kFloatExponentBias = sizeof(F) == 4 ? kFloat32ExponentBias : kFloat64ExponentBias; Label done; { UseScratchRegisterScope temps2(this); Register scratch = temps2.Acquire(); // extract exponent value of the source floating-point to scratch if (std::is_same::value) { fmv_x_d(scratch, src); } else { fmv_x_w(scratch, src); } ExtractBits(scratch2, scratch, kFloatMantissaBits, kFloatExponentBits); } // if src is NaN/+-Infinity/+-Zero or if the exponent is larger than # of bits // in mantissa, the result is the same as src, so move src to dest (to avoid // generating another branch) if (dst != src) { if (std::is_same::value) { fmv_d(dst, src); } else { fmv_s(dst, src); } } { Label not_NaN; UseScratchRegisterScope temps2(this); Register scratch = temps2.Acquire(); // According to the wasm spec // (https://webassembly.github.io/spec/core/exec/numerics.html#aux-nans) // if input is canonical NaN, then output is canonical NaN, and if input is // any other NaN, then output is any NaN with most significant bit of // payload is 1. In RISC-V, feq_d will set scratch to 0 if src is a NaN. If // src is not a NaN, branch to the label and do nothing, but if it is, // fmin_d will set dst to the canonical NaN. if (std::is_same::value) { feq_d(scratch, src, src); bnez(scratch, ¬_NaN); fmin_d(dst, src, src); } else { feq_s(scratch, src, src); bnez(scratch, ¬_NaN); fmin_s(dst, src, src); } bind(¬_NaN); } // If real exponent (i.e., scratch2 - kFloatExponentBias) is greater than // kFloat32MantissaBits, it means the floating-point value has no fractional // part, thus the input is already rounded, jump to done. Note that, NaN and // Infinity in floating-point representation sets maximal exponent value, so // they also satisfy (scratch2 - kFloatExponentBias >= kFloatMantissaBits), // and JS round semantics specify that rounding of NaN (Infinity) returns NaN // (Infinity), so NaN and Infinity are considered rounded value too. ma_branch(&done, GreaterThanOrEqual, scratch2, Operand(kFloatExponentBias + kFloatMantissaBits)); // Actual rounding is needed along this path // old_src holds the original input, needed for the case of src == dst FPURegister old_src = src; if (src == dst) { MOZ_ASSERT(fpu_scratch != dst); fmv_d(fpu_scratch, src); old_src = fpu_scratch; } // Since only input whose real exponent value is less than kMantissaBits // (i.e., 23 or 52-bits) falls into this path, the value range of the input // falls into that of 23- or 53-bit integers. So we round the input to integer // values, then convert them back to floating-point. { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); if (std::is_same::value) { fcvt_l_d(scratch, src, frm); fcvt_d_l(dst, scratch, frm); } else { fcvt_w_s(scratch, src, frm); fcvt_s_w(dst, scratch, frm); } } // A special handling is needed if the input is a very small positive/negative // number that rounds to zero. JS semantics requires that the rounded result // retains the sign of the input, so a very small positive (negative) // floating-point number should be rounded to positive (negative) 0. // Therefore, we use sign-bit injection to produce +/-0 correctly. Instead of // testing for zero w/ a branch, we just insert sign-bit for everyone on this // path (this is where old_src is needed) if (std::is_same::value) { fsgnj_d(dst, dst, old_src); } else { fsgnj_s(dst, dst, old_src); } bind(&done); } template void MacroAssemblerRiscv64::RoundFloatingPointToInteger(Register rd, FPURegister fs, Register result, CvtFunc fcvt_generator, bool Inexact) { // Save csr_fflags to scratch & clear exception flags if (result != Register::Invalid()) { BlockTrampolinePoolScope block_trampoline_pool(this, 6); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); int exception_flags = kInvalidOperation; if (Inexact) exception_flags |= kInexact; csrrci(scratch, csr_fflags, exception_flags); // actual conversion instruction fcvt_generator(this, rd, fs); // check kInvalidOperation flag (out-of-range, NaN) // set result to 1 if normal, otherwise set result to 0 for abnormal frflags(result); andi(result, result, exception_flags); seqz(result, result); // result <-- 1 (normal), result <-- 0 (abnormal) // restore csr_fflags csrw(csr_fflags, scratch); } else { // actual conversion instruction fcvt_generator(this, rd, fs); } } void MacroAssemblerRiscv64::Trunc_uw_d(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_wu_d(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Trunc_w_d(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_d(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Trunc_uw_s(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_wu_s(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Trunc_w_s(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_s(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Trunc_ul_d(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_lu_d(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Trunc_l_d(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_l_d(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Trunc_ul_s(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_lu_s(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Trunc_l_s(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_l_s(dst, src, RTZ); }, Inexact); } void MacroAssemblerRiscv64::Floor_d_d(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RDN); } void MacroAssemblerRiscv64::Ceil_d_d(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RUP); } void MacroAssemblerRiscv64::Trunc_d_d(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RTZ); } void MacroAssemblerRiscv64::Round_d_d(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RNE); } void MacroAssemblerRiscv64::Floor_s_s(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RDN); } void MacroAssemblerRiscv64::Ceil_s_s(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RUP); } void MacroAssemblerRiscv64::Trunc_s_s(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RTZ); } void MacroAssemblerRiscv64::Round_s_s(FPURegister dst, FPURegister src, FPURegister fpu_scratch) { RoundHelper(dst, src, fpu_scratch, RNE); } void MacroAssemblerRiscv64::Round_w_s(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_s(dst, src, RNE); }, Inexact); } void MacroAssemblerRiscv64::Round_w_d(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_d(dst, src, RNE); }, Inexact); } void MacroAssemblerRiscv64::Ceil_w_s(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_s(dst, src, RUP); }, Inexact); } void MacroAssemblerRiscv64::Ceil_w_d(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_d(dst, src, RUP); }, Inexact); } void MacroAssemblerRiscv64::Floor_w_s(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_s(dst, src, RDN); }, Inexact); } void MacroAssemblerRiscv64::Floor_w_d(Register rd, FPURegister fs, Register result, bool Inexact) { RoundFloatingPointToInteger( rd, fs, result, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_d(dst, src, RDN); }, Inexact); } // Checks whether a double is representable as a 32-bit integer. If so, the // integer is written to the output register. Otherwise, a bailout is taken to // the given snapshot. This function overwrites the scratch float register. void MacroAssemblerRiscv64Compat::convertDoubleToInt32(FloatRegister src, Register dest, Label* fail, bool negativeZeroCheck) { if (negativeZeroCheck) { fclass_d(dest, src); ma_b(dest, Imm32(kNegativeZero), fail, Equal); } UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Trunc_w_d(dest, src, scratch, true); ma_b(scratch, Imm32(0), fail, Equal); } void MacroAssemblerRiscv64Compat::convertDoubleToPtr(FloatRegister src, Register dest, Label* fail, bool negativeZeroCheck) { if (negativeZeroCheck) { fclass_d(dest, src); ma_b(dest, Imm32(kNegativeZero), fail, Equal); } UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Trunc_l_d(dest, src, scratch, true); ma_b(scratch, Imm32(0), fail, Equal); } // Checks whether a float32 is representable as a 32-bit integer. If so, the // integer is written to the output register. Otherwise, a bailout is taken to // the given snapshot. This function overwrites the scratch float register. void MacroAssemblerRiscv64Compat::convertFloat32ToInt32( FloatRegister src, Register dest, Label* fail, bool negativeZeroCheck) { if (negativeZeroCheck) { fclass_d(dest, src); ma_b(dest, Imm32(kNegativeZero), fail, Equal); } UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Trunc_w_s(dest, src, scratch, true); ma_b(scratch, Imm32(0), fail, Equal); } void MacroAssemblerRiscv64Compat::convertFloat32ToDouble(FloatRegister src, FloatRegister dest) { fcvt_d_s(dest, src); } void MacroAssemblerRiscv64Compat::convertInt32ToFloat32(Register src, FloatRegister dest) { fcvt_s_w(dest, src); } void MacroAssemblerRiscv64Compat::convertInt32ToFloat32(const Address& src, FloatRegister dest) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); load32(src, scratch); fcvt_s_w(dest, scratch); } void MacroAssemblerRiscv64Compat::movq(Register rj, Register rd) { mv(rd, rj); } // Memory. FaultingCodeOffset MacroAssemblerRiscv64::ma_loadDouble(FloatRegister dest, Address address) { int16_t encodedOffset; Register base; if (!is_int12(address.offset)) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, Imm32(address.offset)); add(ScratchRegister, address.base, ScratchRegister); base = ScratchRegister; encodedOffset = 0; } else { encodedOffset = address.offset; base = address.base; } FaultingCodeOffset fco = FaultingCodeOffset(currentOffset()); fld(dest, base, encodedOffset); return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_loadFloat(FloatRegister dest, Address address) { int16_t encodedOffset; Register base; if (!is_int12(address.offset)) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, Imm32(address.offset)); add(ScratchRegister, address.base, ScratchRegister); base = ScratchRegister; encodedOffset = 0; } else { encodedOffset = address.offset; base = address.base; } FaultingCodeOffset fco = FaultingCodeOffset(currentOffset()); flw(dest, base, encodedOffset); return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_load( Register dest, Address address, LoadStoreSize size, LoadStoreExtension extension) { int16_t encodedOffset; Register base; if (!is_int12(address.offset)) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, Imm32(address.offset)); add(ScratchRegister, address.base, ScratchRegister); base = ScratchRegister; encodedOffset = 0; } else { encodedOffset = address.offset; base = address.base; } FaultingCodeOffset fco = FaultingCodeOffset(currentOffset()); switch (size) { case SizeByte: if (ZeroExtend == extension) { lbu(dest, base, encodedOffset); } else { lb(dest, base, encodedOffset); } break; case SizeHalfWord: if (ZeroExtend == extension) { lhu(dest, base, encodedOffset); } else { lh(dest, base, encodedOffset); } break; case SizeWord: if (ZeroExtend == extension) { lwu(dest, base, encodedOffset); } else { lw(dest, base, encodedOffset); } break; case SizeDouble: ld(dest, base, encodedOffset); break; default: MOZ_CRASH("Invalid argument for ma_load"); } return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_store( Register data, const BaseIndex& dest, LoadStoreSize size, LoadStoreExtension extension) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); asMasm().computeScaledAddress(dest, scratch2); return asMasm().ma_store(data, Address(scratch2, dest.offset), size, extension); } FaultingCodeOffset MacroAssemblerRiscv64::ma_store( Imm32 imm, const BaseIndex& dest, LoadStoreSize size, LoadStoreExtension extension) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Register address = temps.Acquire(); // Make sure that scratch contains absolute address so that // offset is 0. computeScaledAddress(dest, address); // Scrach register is free now, use it for loading imm value ma_li(scratch, imm); // with offset=0 ScratchRegister will not be used in ma_store() // so we can use it as a parameter here return ma_store(scratch, Address(address, 0), size, extension); } FaultingCodeOffset MacroAssemblerRiscv64::ma_store( Imm32 imm, Address address, LoadStoreSize size, LoadStoreExtension extension) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, imm); return ma_store(scratch, address, size, extension); } FaultingCodeOffset MacroAssemblerRiscv64::ma_store( Register data, Address address, LoadStoreSize size, LoadStoreExtension extension) { int16_t encodedOffset; Register base; if (!is_int12(address.offset)) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, Imm32(address.offset)); add(ScratchRegister, address.base, ScratchRegister); base = ScratchRegister; encodedOffset = 0; } else { encodedOffset = address.offset; base = address.base; } FaultingCodeOffset fco = FaultingCodeOffset(currentOffset()); switch (size) { case SizeByte: sb(data, base, encodedOffset); break; case SizeHalfWord: sh(data, base, encodedOffset); break; case SizeWord: sw(data, base, encodedOffset); break; case SizeDouble: sd(data, base, encodedOffset); break; default: MOZ_CRASH("Invalid argument for ma_store"); } return fco; } // Memory. void MacroAssemblerRiscv64::ma_storeDouble(FloatRegister dest, Address address) { int16_t encodedOffset; Register base; if (!is_int12(address.offset)) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, Imm32(address.offset)); add(ScratchRegister, address.base, ScratchRegister); base = ScratchRegister; encodedOffset = 0; } else { encodedOffset = address.offset; base = address.base; } fsd(dest, base, encodedOffset); } void MacroAssemblerRiscv64::ma_storeFloat(FloatRegister dest, Address address) { int16_t encodedOffset; Register base; if (!is_int12(address.offset)) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, Imm32(address.offset)); add(ScratchRegister, address.base, ScratchRegister); base = ScratchRegister; encodedOffset = 0; } else { encodedOffset = address.offset; base = address.base; } fsw(dest, base, encodedOffset); } void MacroAssemblerRiscv64::computeScaledAddress(const BaseIndex& address, Register dest) { Register base = address.base; Register index = address.index; int32_t shift = Imm32::ShiftOf(address.scale).value; UseScratchRegisterScope temps(this); Register tmp = dest == base ? temps.Acquire() : dest; if (shift) { MOZ_ASSERT(shift <= 4); slli(tmp, index, shift); add(dest, base, tmp); } else { add(dest, base, index); } } void MacroAssemblerRiscv64Compat::wasmLoadI64Impl( const wasm::MemoryAccessDesc& access, Register memoryBase, Register ptr, Register ptrScratch, Register64 output, Register tmp) { access.assertOffsetInGuardPages(); uint32_t offset = access.offset(); MOZ_ASSERT_IF(offset, ptrScratch != InvalidReg); // Maybe add the offset. if (offset) { asMasm().addPtr(ImmWord(offset), ptrScratch); ptr = ptrScratch; } asMasm().memoryBarrierBefore(access.sync()); UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); FaultingCodeOffset fco; switch (access.type()) { case Scalar::Int8: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lb(output.reg, ScratchRegister, 0); break; case Scalar::Uint8: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lbu(output.reg, ScratchRegister, 0); break; case Scalar::Int16: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lh(output.reg, ScratchRegister, 0); break; case Scalar::Uint16: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lhu(output.reg, ScratchRegister, 0); break; case Scalar::Int32: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lw(output.reg, ScratchRegister, 0); break; case Scalar::Uint32: // TODO(riscv): Why need zero-extension here? add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lwu(output.reg, ScratchRegister, 0); break; case Scalar::Int64: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); ld(output.reg, ScratchRegister, 0); break; default: MOZ_CRASH("unexpected array type"); } asMasm().append(access, js::wasm::TrapMachineInsnForLoad(access.byteSize()), fco); asMasm().memoryBarrierAfter(access.sync()); } void MacroAssemblerRiscv64Compat::wasmStoreI64Impl( const wasm::MemoryAccessDesc& access, Register64 value, Register memoryBase, Register ptr, Register ptrScratch, Register tmp) { access.assertOffsetInGuardPages(); uint32_t offset = access.offset(); MOZ_ASSERT_IF(offset, ptrScratch != InvalidReg); // Maybe add the offset. if (offset) { asMasm().addPtr(ImmWord(offset), ptrScratch); ptr = ptrScratch; } asMasm().memoryBarrierBefore(access.sync()); UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); FaultingCodeOffset fco; switch (access.type()) { case Scalar::Int8: case Scalar::Uint8: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); sb(value.reg, ScratchRegister, 0); break; case Scalar::Int16: case Scalar::Uint16: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); sh(value.reg, ScratchRegister, 0); break; case Scalar::Int32: case Scalar::Uint32: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); sw(value.reg, ScratchRegister, 0); break; case Scalar::Int64: add(ScratchRegister, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); sd(value.reg, ScratchRegister, 0); break; default: MOZ_CRASH("unexpected array type"); } asMasm().append(access, js::wasm::TrapMachineInsnForLoad(access.byteSize()), fco); asMasm().memoryBarrierAfter(access.sync()); } void MacroAssemblerRiscv64Compat::profilerEnterFrame(Register framePtr, Register scratch) { asMasm().loadJSContext(scratch); loadPtr(Address(scratch, offsetof(JSContext, profilingActivation_)), scratch); storePtr(framePtr, Address(scratch, JitActivation::offsetOfLastProfilingFrame())); storePtr(ImmPtr(nullptr), Address(scratch, JitActivation::offsetOfLastProfilingCallSite())); } void MacroAssemblerRiscv64Compat::profilerExitFrame() { jump(asMasm().runtime()->jitRuntime()->getProfilerExitFrameTail()); } void MacroAssemblerRiscv64Compat::move32(Imm32 imm, Register dest) { ma_li(dest, imm); } void MacroAssemblerRiscv64Compat::move32(Register src, Register dest) { slliw(dest, src, 0); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load8ZeroExtend( const Address& address, Register dest) { return ma_load(dest, address, SizeByte, ZeroExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load8ZeroExtend( const BaseIndex& src, Register dest) { return ma_load(dest, src, SizeByte, ZeroExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load8SignExtend( const Address& address, Register dest) { return ma_load(dest, address, SizeByte, SignExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load8SignExtend( const BaseIndex& src, Register dest) { return ma_load(dest, src, SizeByte, SignExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load16ZeroExtend( const Address& address, Register dest) { return ma_load(dest, address, SizeHalfWord, ZeroExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load16ZeroExtend( const BaseIndex& src, Register dest) { return ma_load(dest, src, SizeHalfWord, ZeroExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load16SignExtend( const Address& address, Register dest) { return ma_load(dest, address, SizeHalfWord, SignExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load16SignExtend( const BaseIndex& src, Register dest) { return ma_load(dest, src, SizeHalfWord, SignExtend); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load32(const Address& address, Register dest) { return ma_load(dest, address, SizeWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load32(const BaseIndex& address, Register dest) { return ma_load(dest, address, SizeWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load32(AbsoluteAddress address, Register dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); movePtr(ImmPtr(address.addr), ScratchRegister); return load32(Address(ScratchRegister, 0), dest); } FaultingCodeOffset MacroAssemblerRiscv64Compat::load32( wasm::SymbolicAddress address, Register dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); movePtr(address, ScratchRegister); return load32(Address(ScratchRegister, 0), dest); } FaultingCodeOffset MacroAssemblerRiscv64Compat::loadPtr(const Address& address, Register dest) { return ma_load(dest, address, SizeDouble); } FaultingCodeOffset MacroAssemblerRiscv64Compat::loadPtr(const BaseIndex& src, Register dest) { return ma_load(dest, src, SizeDouble); } FaultingCodeOffset MacroAssemblerRiscv64Compat::loadPtr(AbsoluteAddress address, Register dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); movePtr(ImmPtr(address.addr), ScratchRegister); return loadPtr(Address(ScratchRegister, 0), dest); } FaultingCodeOffset MacroAssemblerRiscv64Compat::loadPtr( wasm::SymbolicAddress address, Register dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); movePtr(address, ScratchRegister); return loadPtr(Address(ScratchRegister, 0), dest); } FaultingCodeOffset MacroAssemblerRiscv64Compat::loadPrivate( const Address& address, Register dest) { return loadPtr(address, dest); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store8(Imm32 imm, const Address& address) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, imm); return ma_store(ScratchRegister, address, SizeByte); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store8(Register src, const Address& address) { return ma_store(src, address, SizeByte); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store8(Imm32 imm, const BaseIndex& dest) { return ma_store(imm, dest, SizeByte); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store8(Register src, const BaseIndex& dest) { return ma_store(src, dest, SizeByte); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store16( Imm32 imm, const Address& address) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, imm); return ma_store(ScratchRegister, address, SizeHalfWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store16( Register src, const Address& address) { return ma_store(src, address, SizeHalfWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store16(Imm32 imm, const BaseIndex& dest) { return ma_store(imm, dest, SizeHalfWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store16( Register src, const BaseIndex& address) { return ma_store(src, address, SizeHalfWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store32( Register src, AbsoluteAddress address) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); movePtr(ImmPtr(address.addr), ScratchRegister); return store32(src, Address(ScratchRegister, 0)); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store32( Register src, const Address& address) { return ma_store(src, address, SizeWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store32( Imm32 src, const Address& address) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); move32(src, ScratchRegister); return ma_store(ScratchRegister, address, SizeWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store32(Imm32 imm, const BaseIndex& dest) { return ma_store(imm, dest, SizeWord); } FaultingCodeOffset MacroAssemblerRiscv64Compat::store32(Register src, const BaseIndex& dest) { return ma_store(src, dest, SizeWord); } template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr(ImmWord imm, T address) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, imm); return ma_store(ScratchRegister, address, SizeDouble); } template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr
( ImmWord imm, Address address); template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr( ImmWord imm, BaseIndex address); template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr(ImmPtr imm, T address) { return storePtr(ImmWord(uintptr_t(imm.value)), address); } template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr
( ImmPtr imm, Address address); template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr( ImmPtr imm, BaseIndex address); template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr(ImmGCPtr imm, T address) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); movePtr(imm, ScratchRegister); return storePtr(ScratchRegister, address); } template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr
( ImmGCPtr imm, Address address); template FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr( ImmGCPtr imm, BaseIndex address); FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr( Register src, const Address& address) { return ma_store(src, address, SizeDouble); } FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr( Register src, const BaseIndex& address) { return ma_store(src, address, SizeDouble); } FaultingCodeOffset MacroAssemblerRiscv64Compat::storePtr(Register src, AbsoluteAddress dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); movePtr(ImmPtr(dest.addr), ScratchRegister); return storePtr(src, Address(ScratchRegister, 0)); } void MacroAssemblerRiscv64Compat::testNullSet(Condition cond, const ValueOperand& value, Register dest) { MOZ_ASSERT(cond == Equal || cond == NotEqual); UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); splitTag(value, ScratchRegister); ma_cmp_set(dest, ScratchRegister, ImmTag(JSVAL_TAG_NULL), cond); } void MacroAssemblerRiscv64Compat::testObjectSet(Condition cond, const ValueOperand& value, Register dest) { MOZ_ASSERT(cond == Equal || cond == NotEqual); UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); splitTag(value, ScratchRegister); ma_cmp_set(dest, ScratchRegister, ImmTag(JSVAL_TAG_OBJECT), cond); } void MacroAssemblerRiscv64Compat::testUndefinedSet(Condition cond, const ValueOperand& value, Register dest) { MOZ_ASSERT(cond == Equal || cond == NotEqual); UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); splitTag(value, ScratchRegister); ma_cmp_set(dest, ScratchRegister, ImmTag(JSVAL_TAG_UNDEFINED), cond); } void MacroAssemblerRiscv64Compat::unboxInt32(const ValueOperand& operand, Register dest) { slliw(dest, operand.valueReg(), 0); } void MacroAssemblerRiscv64Compat::unboxInt32(Register src, Register dest) { slliw(dest, src, 0); } void MacroAssemblerRiscv64Compat::unboxInt32(const Address& src, Register dest) { load32(Address(src.base, src.offset), dest); } void MacroAssemblerRiscv64Compat::unboxInt32(const BaseIndex& src, Register dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); computeScaledAddress(src, ScratchRegister); load32(Address(ScratchRegister, src.offset), dest); } void MacroAssemblerRiscv64Compat::unboxBoolean(const ValueOperand& operand, Register dest) { ExtractBits(dest, operand.valueReg(), 0, 32); } void MacroAssemblerRiscv64Compat::unboxBoolean(Register src, Register dest) { ExtractBits(dest, src, 0, 32); } void MacroAssemblerRiscv64Compat::unboxBoolean(const Address& src, Register dest) { ma_load(dest, Address(src.base, src.offset), SizeWord, ZeroExtend); } void MacroAssemblerRiscv64Compat::unboxBoolean(const BaseIndex& src, Register dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); computeScaledAddress(src, ScratchRegister); ma_load(dest, Address(ScratchRegister, src.offset), SizeWord, ZeroExtend); } void MacroAssemblerRiscv64Compat::unboxDouble(const ValueOperand& operand, FloatRegister dest) { fmv_d_x(dest, operand.valueReg()); } void MacroAssemblerRiscv64Compat::unboxDouble(const Address& src, FloatRegister dest) { ma_loadDouble(dest, Address(src.base, src.offset)); } void MacroAssemblerRiscv64Compat::unboxDouble(const BaseIndex& src, FloatRegister dest) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); loadPtr(src, scratch); unboxDouble(ValueOperand(scratch), dest); } void MacroAssemblerRiscv64Compat::unboxString(const ValueOperand& operand, Register dest) { unboxNonDouble(operand, dest, JSVAL_TYPE_STRING); } void MacroAssemblerRiscv64Compat::unboxString(Register src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_STRING); } void MacroAssemblerRiscv64Compat::unboxString(const Address& src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_STRING); } void MacroAssemblerRiscv64Compat::unboxSymbol(const ValueOperand& operand, Register dest) { unboxNonDouble(operand, dest, JSVAL_TYPE_SYMBOL); } void MacroAssemblerRiscv64Compat::unboxSymbol(Register src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_SYMBOL); } void MacroAssemblerRiscv64Compat::unboxSymbol(const Address& src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_SYMBOL); } void MacroAssemblerRiscv64Compat::unboxBigInt(const ValueOperand& operand, Register dest) { unboxNonDouble(operand, dest, JSVAL_TYPE_BIGINT); } void MacroAssemblerRiscv64Compat::unboxBigInt(Register src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_BIGINT); } void MacroAssemblerRiscv64Compat::unboxBigInt(const Address& src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_BIGINT); } void MacroAssemblerRiscv64Compat::unboxObject(const ValueOperand& src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_OBJECT); } void MacroAssemblerRiscv64Compat::unboxObject(Register src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_OBJECT); } void MacroAssemblerRiscv64Compat::unboxObject(const Address& src, Register dest) { unboxNonDouble(src, dest, JSVAL_TYPE_OBJECT); } void MacroAssemblerRiscv64Compat::unboxValue(const ValueOperand& src, AnyRegister dest, JSValueType type) { if (dest.isFloat()) { Label notInt32, end; asMasm().branchTestInt32(Assembler::NotEqual, src, ¬Int32); convertInt32ToDouble(src.valueReg(), dest.fpu()); ma_branch(&end); bind(¬Int32); unboxDouble(src, dest.fpu()); bind(&end); } else { unboxNonDouble(src, dest.gpr(), type); } } void MacroAssemblerRiscv64Compat::boxDouble(FloatRegister src, const ValueOperand& dest, FloatRegister) { fmv_x_d(dest.valueReg(), src); } void MacroAssemblerRiscv64Compat::boxNonDouble(JSValueType type, Register src, const ValueOperand& dest) { MOZ_ASSERT(src != dest.valueReg()); boxValue(type, src, dest.valueReg()); } void MacroAssemblerRiscv64Compat::boolValueToDouble(const ValueOperand& operand, FloatRegister dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); convertBoolToInt32(operand.valueReg(), ScratchRegister); convertInt32ToDouble(ScratchRegister, dest); } void MacroAssemblerRiscv64Compat::int32ValueToDouble( const ValueOperand& operand, FloatRegister dest) { convertInt32ToDouble(operand.valueReg(), dest); } void MacroAssemblerRiscv64Compat::boolValueToFloat32( const ValueOperand& operand, FloatRegister dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); convertBoolToInt32(operand.valueReg(), ScratchRegister); convertInt32ToFloat32(ScratchRegister, dest); } void MacroAssemblerRiscv64Compat::int32ValueToFloat32( const ValueOperand& operand, FloatRegister dest) { convertInt32ToFloat32(operand.valueReg(), dest); } void MacroAssemblerRiscv64Compat::loadConstantFloat32(float f, FloatRegister dest) { ma_lis(dest, f); } void MacroAssemblerRiscv64Compat::loadInt32OrDouble(const Address& src, FloatRegister dest) { Label notInt32, end; // If it's an int, convert it to double. UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Register SecondScratchReg = temps.Acquire(); loadPtr(Address(src.base, src.offset), ScratchRegister); srli(SecondScratchReg, ScratchRegister, JSVAL_TAG_SHIFT); asMasm().branchTestInt32(Assembler::NotEqual, SecondScratchReg, ¬Int32); loadPtr(Address(src.base, src.offset), SecondScratchReg); convertInt32ToDouble(SecondScratchReg, dest); ma_branch(&end); // Not an int, just load as double. bind(¬Int32); unboxDouble(src, dest); bind(&end); } void MacroAssemblerRiscv64Compat::loadInt32OrDouble(const BaseIndex& addr, FloatRegister dest) { Label notInt32, end; UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Register SecondScratchReg = temps.Acquire(); // If it's an int, convert it to double. computeScaledAddress(addr, SecondScratchReg); // Since we only have one scratch, we need to stomp over it with the tag. loadPtr(Address(SecondScratchReg, 0), ScratchRegister); srli(SecondScratchReg, ScratchRegister, JSVAL_TAG_SHIFT); asMasm().branchTestInt32(Assembler::NotEqual, SecondScratchReg, ¬Int32); computeScaledAddress(addr, SecondScratchReg); loadPtr(Address(SecondScratchReg, 0), SecondScratchReg); convertInt32ToDouble(SecondScratchReg, dest); ma_branch(&end); // Not an int, just load as double. bind(¬Int32); // First, recompute the offset that had been stored in the scratch register // since the scratch register was overwritten loading in the type. computeScaledAddress(addr, SecondScratchReg); unboxDouble(Address(SecondScratchReg, 0), dest); bind(&end); } void MacroAssemblerRiscv64Compat::loadConstantDouble(double dp, FloatRegister dest) { ma_lid(dest, dp); } Register MacroAssemblerRiscv64Compat::extractObject(const Address& address, Register scratch) { loadPtr(Address(address.base, address.offset), scratch); ExtractBits(scratch, scratch, 0, JSVAL_TAG_SHIFT); return scratch; } Register MacroAssemblerRiscv64Compat::extractTag(const Address& address, Register scratch) { loadPtr(Address(address.base, address.offset), scratch); ExtractBits(scratch, scratch, JSVAL_TAG_SHIFT, 64 - JSVAL_TAG_SHIFT); return scratch; } Register MacroAssemblerRiscv64Compat::extractTag(const BaseIndex& address, Register scratch) { computeScaledAddress(address, scratch); return extractTag(Address(scratch, address.offset), scratch); } ///////////////////////////////////////////////////////////////// // X86/X64-common/ARM/LoongArch interface. ///////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////// // X86/X64-common/ARM/MIPS interface. ///////////////////////////////////////////////////////////////// void MacroAssemblerRiscv64Compat::storeValue(ValueOperand val, const BaseIndex& dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); computeScaledAddress(dest, ScratchRegister); storeValue(val, Address(ScratchRegister, dest.offset)); } void MacroAssemblerRiscv64Compat::storeValue(JSValueType type, Register reg, BaseIndex dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); computeScaledAddress(dest, ScratchRegister); int32_t offset = dest.offset; if (!is_int12(offset)) { UseScratchRegisterScope temps(this); Register SecondScratchReg = temps.Acquire(); ma_li(SecondScratchReg, Imm32(offset)); add(ScratchRegister, ScratchRegister, SecondScratchReg); offset = 0; } storeValue(type, reg, Address(ScratchRegister, offset)); } void MacroAssemblerRiscv64Compat::storeValue(ValueOperand val, const Address& dest) { storePtr(val.valueReg(), Address(dest.base, dest.offset)); } void MacroAssemblerRiscv64Compat::storeValue(JSValueType type, Register reg, Address dest) { if (type == JSVAL_TYPE_INT32 || type == JSVAL_TYPE_BOOLEAN) { store32(reg, dest); JSValueShiftedTag tag = (JSValueShiftedTag)JSVAL_TYPE_TO_SHIFTED_TAG(type); store32(((Imm64(tag)).secondHalf()), Address(dest.base, dest.offset + 4)); } else { ScratchRegisterScope SecondScratchReg(asMasm()); MOZ_ASSERT(dest.base != SecondScratchReg); ma_li(SecondScratchReg, ImmTag(JSVAL_TYPE_TO_TAG(type))); slli(SecondScratchReg, SecondScratchReg, JSVAL_TAG_SHIFT); InsertBits(SecondScratchReg, reg, 0, JSVAL_TAG_SHIFT); storePtr(SecondScratchReg, Address(dest.base, dest.offset)); } } void MacroAssemblerRiscv64Compat::storeValue(const Value& val, Address dest) { UseScratchRegisterScope temps(this); Register SecondScratchReg = temps.Acquire(); if (val.isGCThing()) { writeDataRelocation(val); movWithPatch(ImmWord(val.asRawBits()), SecondScratchReg); } else { ma_li(SecondScratchReg, ImmWord(val.asRawBits())); } storePtr(SecondScratchReg, Address(dest.base, dest.offset)); } void MacroAssemblerRiscv64Compat::storeValue(const Value& val, BaseIndex dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Register SecondScratchReg = temps.Acquire(); computeScaledAddress(dest, ScratchRegister); int32_t offset = dest.offset; if (!is_int12(offset)) { ma_li(SecondScratchReg, Imm32(offset)); add(ScratchRegister, ScratchRegister, SecondScratchReg); offset = 0; } storeValue(val, Address(ScratchRegister, offset)); } void MacroAssemblerRiscv64Compat::loadValue(const BaseIndex& addr, ValueOperand val) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); computeScaledAddress(addr, ScratchRegister); loadValue(Address(ScratchRegister, addr.offset), val); } void MacroAssemblerRiscv64Compat::loadValue(Address src, ValueOperand val) { loadPtr(Address(src.base, src.offset), val.valueReg()); } void MacroAssemblerRiscv64Compat::tagValue(JSValueType type, Register payload, ValueOperand dest) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); MOZ_ASSERT(dest.valueReg() != ScratchRegister); JitSpew(JitSpew_Codegen, "[ tagValue"); if (payload != dest.valueReg()) { mv(dest.valueReg(), payload); } ma_li(ScratchRegister, ImmTag(JSVAL_TYPE_TO_TAG(type))); InsertBits(dest.valueReg(), ScratchRegister, JSVAL_TAG_SHIFT, 64 - JSVAL_TAG_SHIFT); if (type == JSVAL_TYPE_INT32 || type == JSVAL_TYPE_BOOLEAN) { InsertBits(dest.valueReg(), zero, 32, JSVAL_TAG_SHIFT - 32); } JitSpew(JitSpew_Codegen, "]"); } void MacroAssemblerRiscv64Compat::pushValue(ValueOperand val) { // Allocate stack slots for Value. One for each. asMasm().subPtr(Imm32(sizeof(Value)), StackPointer); // Store Value storeValue(val, Address(StackPointer, 0)); } void MacroAssemblerRiscv64Compat::pushValue(const Address& addr) { // Load value before allocate stack, addr.base may be is sp. UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); loadPtr(Address(addr.base, addr.offset), ScratchRegister); ma_sub64(StackPointer, StackPointer, Imm32(sizeof(Value))); storePtr(ScratchRegister, Address(StackPointer, 0)); } void MacroAssemblerRiscv64Compat::popValue(ValueOperand val) { ld(val.valueReg(), StackPointer, 0); ma_add64(StackPointer, StackPointer, Imm32(sizeof(Value))); } void MacroAssemblerRiscv64Compat::breakpoint(uint32_t value) { break_(value); } void MacroAssemblerRiscv64Compat::ensureDouble(const ValueOperand& source, FloatRegister dest, Label* failure) { Label isDouble, done; { ScratchTagScope tag(asMasm(), source); splitTagForTest(source, tag); asMasm().branchTestDouble(Assembler::Equal, tag, &isDouble); asMasm().branchTestInt32(Assembler::NotEqual, tag, failure); } UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); unboxInt32(source, ScratchRegister); convertInt32ToDouble(ScratchRegister, dest); jump(&done); bind(&isDouble); unboxDouble(source, dest); bind(&done); } void MacroAssemblerRiscv64Compat::handleFailureWithHandlerTail( Label* profilerExitTail, Label* bailoutTail) { // Reserve space for exception information. int size = (sizeof(ResumeFromException) + ABIStackAlignment) & ~(ABIStackAlignment - 1); asMasm().subPtr(Imm32(size), StackPointer); mv(a0, StackPointer); // Use a0 since it is a first function argument // Call the handler. using Fn = void (*)(ResumeFromException* rfe); asMasm().setupUnalignedABICall(a1); asMasm().passABIArg(a0); asMasm().callWithABI( ABIType::General, CheckUnsafeCallWithABI::DontCheckHasExitFrame); Label entryFrame; Label catch_; Label finally; Label returnBaseline; Label returnIon; Label bailout; Label wasm; Label wasmCatch; // Already clobbered a0, so use it... load32(Address(StackPointer, ResumeFromException::offsetOfKind()), a0); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::EntryFrame), &entryFrame); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::Catch), &catch_); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::Finally), &finally); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::ForcedReturnBaseline), &returnBaseline); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::ForcedReturnIon), &returnIon); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::Bailout), &bailout); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::Wasm), &wasm); asMasm().branch32(Assembler::Equal, a0, Imm32(ExceptionResumeKind::WasmCatch), &wasmCatch); breakpoint(); // Invalid kind. // No exception handler. Load the error value, restore state and return from // the entry frame. bind(&entryFrame); asMasm().moveValue(MagicValue(JS_ION_ERROR), JSReturnOperand); loadPtr(Address(StackPointer, ResumeFromException::offsetOfFramePointer()), FramePointer); loadPtr(Address(StackPointer, ResumeFromException::offsetOfStackPointer()), StackPointer); // We're going to be returning by the ion calling convention ma_pop(ra); jump(ra); nop(); // If we found a catch handler, this must be a baseline frame. Restore // state and jump to the catch block. bind(&catch_); loadPtr(Address(StackPointer, ResumeFromException::offsetOfTarget()), a0); loadPtr(Address(StackPointer, ResumeFromException::offsetOfFramePointer()), FramePointer); loadPtr(Address(StackPointer, ResumeFromException::offsetOfStackPointer()), StackPointer); jump(a0); // If we found a finally block, this must be a baseline frame. Push three // values expected by the finally block: the exception, the exception stack, // and BooleanValue(true). bind(&finally); ValueOperand exception = ValueOperand(a1); loadValue(Address(sp, ResumeFromException::offsetOfException()), exception); ValueOperand exceptionStack = ValueOperand(a2); loadValue(Address(sp, ResumeFromException::offsetOfExceptionStack()), exceptionStack); loadPtr(Address(sp, ResumeFromException::offsetOfTarget()), a0); loadPtr(Address(sp, ResumeFromException::offsetOfFramePointer()), FramePointer); loadPtr(Address(sp, ResumeFromException::offsetOfStackPointer()), sp); pushValue(exception); pushValue(exceptionStack); pushValue(BooleanValue(true)); jump(a0); // Return BaselineFrame->returnValue() to the caller. // Used in debug mode and for GeneratorReturn. Label profilingInstrumentation; bind(&returnBaseline); loadPtr(Address(StackPointer, ResumeFromException::offsetOfFramePointer()), FramePointer); loadPtr(Address(StackPointer, ResumeFromException::offsetOfStackPointer()), StackPointer); loadValue(Address(FramePointer, BaselineFrame::reverseOffsetOfReturnValue()), JSReturnOperand); jump(&profilingInstrumentation); // Return the given value to the caller. bind(&returnIon); loadValue(Address(StackPointer, ResumeFromException::offsetOfException()), JSReturnOperand); loadPtr(Address(StackPointer, ResumeFromException::offsetOfFramePointer()), FramePointer); loadPtr(Address(StackPointer, ResumeFromException::offsetOfStackPointer()), StackPointer); // If profiling is enabled, then update the lastProfilingFrame to refer to // caller frame before returning. This code is shared by ForcedReturnIon // and ForcedReturnBaseline. bind(&profilingInstrumentation); { Label skipProfilingInstrumentation; // Test if profiler enabled. AbsoluteAddress addressOfEnabled( asMasm().runtime()->geckoProfiler().addressOfEnabled()); asMasm().branch32(Assembler::Equal, addressOfEnabled, Imm32(0), &skipProfilingInstrumentation); jump(profilerExitTail); bind(&skipProfilingInstrumentation); } mv(StackPointer, FramePointer); pop(FramePointer); ret(); // If we are bailing out to baseline to handle an exception, jump to // the bailout tail stub. Load 1 (true) in ReturnReg to indicate success. bind(&bailout); loadPtr(Address(sp, ResumeFromException::offsetOfBailoutInfo()), a2); loadPtr(Address(StackPointer, ResumeFromException::offsetOfStackPointer()), StackPointer); ma_li(ReturnReg, Imm32(1)); jump(bailoutTail); // If we are throwing and the innermost frame was a wasm frame, reset SP and // FP; SP is pointing to the unwound return address to the wasm entry, so // we can just ret(). bind(&wasm); loadPtr(Address(StackPointer, ResumeFromException::offsetOfFramePointer()), FramePointer); loadPtr(Address(StackPointer, ResumeFromException::offsetOfStackPointer()), StackPointer); ma_li(InstanceReg, ImmWord(wasm::FailInstanceReg)); ret(); // Found a wasm catch handler, restore state and jump to it. bind(&wasmCatch); loadPtr(Address(sp, ResumeFromException::offsetOfTarget()), a1); loadPtr(Address(StackPointer, ResumeFromException::offsetOfFramePointer()), FramePointer); loadPtr(Address(StackPointer, ResumeFromException::offsetOfStackPointer()), StackPointer); jump(a1); } CodeOffset MacroAssemblerRiscv64Compat::toggledJump(Label* label) { CodeOffset ret(nextOffset().getOffset()); BranchShort(label); return ret; } CodeOffset MacroAssemblerRiscv64Compat::toggledCall(JitCode* target, bool enabled) { DEBUG_PRINTF("\ttoggledCall\n"); UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 8); BufferOffset bo = nextOffset(); CodeOffset offset(bo.getOffset()); addPendingJump(bo, ImmPtr(target->raw()), RelocationKind::JITCODE); ma_liPatchable(ScratchRegister, ImmPtr(target->raw())); if (enabled) { jalr(ScratchRegister); } else { nop(); } MOZ_ASSERT_IF(!oom(), nextOffset().getOffset() - offset.offset() == ToggledCallSize(nullptr)); return offset; } void MacroAssembler::subFromStackPtr(Imm32 imm32) { if (imm32.value) { asMasm().subPtr(imm32, StackPointer); } } void MacroAssembler::clampDoubleToUint8(FloatRegister input, Register output) { JitSpew(JitSpew_Codegen, "[ clampDoubleToUint8"); Label nan, done; UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); feq_d(scratch, input, input); beqz(scratch, &nan); addi(zero, scratch, 0x11); Round_w_d(output, input); clampIntToUint8(output); ma_branch(&done); // Input is nan bind(&nan); mv(output, zero_reg); bind(&done); JitSpew(JitSpew_Codegen, "]"); } //{{{ check_macroassembler_style // =============================================================== // MacroAssembler high-level usage. bool MacroAssembler::convertUInt64ToDoubleNeedsTemp() { return false; } CodeOffset MacroAssembler::call(Label* label) { BranchAndLink(label); return CodeOffset(currentOffset()); } CodeOffset MacroAssembler::call(Register reg) { jalr(reg, 0); return CodeOffset(currentOffset()); } CodeOffset MacroAssembler::call(wasm::SymbolicAddress target) { UseScratchRegisterScope temps(this); temps.Exclude(GeneralRegisterSet(1 << CallReg.code())); movePtr(target, CallReg); return call(CallReg); } CodeOffset MacroAssembler::farJumpWithPatch() { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Register scratch2 = temps.Acquire(); // Allocate space which will be patched by patchFarJump(). CodeOffset farJump(nextInstrOffset(5).getOffset()); auipc(scratch, 0); lw(scratch2, scratch, 4 * sizeof(Instr)); add(scratch, scratch, scratch2); jr(scratch, 0); spew(".space 32bit initValue 0xffff ffff"); emit(UINT32_MAX); return farJump; } CodeOffset MacroAssembler::moveNearAddressWithPatch(Register dest) { return movWithPatch(ImmPtr(nullptr), dest); } CodeOffset MacroAssembler::nopPatchableToCall() { BlockTrampolinePoolScope block_trampoline_pool(this, 7); // riscv64 nop(); // lui(rd, (int32_t)high_20); nop(); // addi(rd, rd, low_12); // 31 bits in rd. nop(); // slli(rd, rd, 11); // Space for next 11 bis nop(); // ori(rd, rd, b11); // 11 bits are put in. 42 bit in rd nop(); // slli(rd, rd, 6); // Space for next 6 bits nop(); // ori(rd, rd, a6); // 6 bits are put in. 48 bis in rd nop(); // jirl return CodeOffset(currentOffset()); } FaultingCodeOffset MacroAssembler::wasmTrapInstruction() { BlockTrampolinePoolScope block_trampoline_pool(this, 2); FaultingCodeOffset fco = FaultingCodeOffset(currentOffset()); illegal_trap(kWasmTrapCode); ebreak(); return fco; } size_t MacroAssembler::PushRegsInMaskSizeInBytes(LiveRegisterSet set) { return set.gprs().size() * sizeof(intptr_t) + set.fpus().getPushSizeInBytes(); } template void MacroAssembler::branchValueIsNurseryCellImpl(Condition cond, const T& value, Register temp, Label* label) { MOZ_ASSERT(cond == Assembler::Equal || cond == Assembler::NotEqual); Label done; branchTestGCThing(Assembler::NotEqual, value, cond == Assembler::Equal ? &done : label); // temp may be InvalidReg, use scratch2 instead. UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); getGCThingValueChunk(value, scratch2); loadPtr(Address(scratch2, gc::ChunkStoreBufferOffset), scratch2); branchPtr(InvertCondition(cond), scratch2, ImmWord(0), label); bind(&done); } template void MacroAssembler::storeUnboxedValue(const ConstantOrRegister& value, MIRType valueType, const T& dest) { MOZ_ASSERT(valueType < MIRType::Value); if (valueType == MIRType::Double) { boxDouble(value.reg().typedReg().fpu(), dest); return; } if (value.constant()) { storeValue(value.value(), dest); } else { storeValue(ValueTypeFromMIRType(valueType), value.reg().typedReg().gpr(), dest); } } template void MacroAssembler::storeUnboxedValue(const ConstantOrRegister& value, MIRType valueType, const Address& dest); template void MacroAssembler::storeUnboxedValue( const ConstantOrRegister& value, MIRType valueType, const BaseObjectElementIndex& dest); // =============================================================== // Jit Frames. uint32_t MacroAssembler::pushFakeReturnAddress(Register scratch) { CodeLabel cl; ma_li(scratch, &cl); Push(scratch); bind(&cl); uint32_t retAddr = currentOffset(); addCodeLabel(cl); return retAddr; } //=============================== // AtomicOp template static void AtomicExchange(MacroAssembler& masm, const wasm::MemoryAccessDesc* access, Scalar::Type type, Synchronization sync, const T& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { ScratchRegisterScope scratch(masm); UseScratchRegisterScope temps(&masm); Register scratch2 = temps.Acquire(); bool signExtend = Scalar::isSignedIntType(type); unsigned nbytes = Scalar::byteSize(type); switch (nbytes) { case 1: case 2: break; case 4: MOZ_ASSERT(valueTemp == InvalidReg); MOZ_ASSERT(offsetTemp == InvalidReg); MOZ_ASSERT(maskTemp == InvalidReg); break; default: MOZ_CRASH(); } Label again; masm.computeEffectiveAddress(mem, scratch); if (nbytes == 4) { masm.memoryBarrierBefore(sync); masm.bind(&again); BlockTrampolinePoolScope block_trampoline_pool(&masm, 5); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, output, scratch); masm.or_(scratch2, value, zero); masm.sc_w(true, true, scratch2, scratch, scratch2); masm.ma_b(scratch2, Register(scratch2), &again, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); return; } masm.andi(offsetTemp, scratch, 3); masm.subPtr(offsetTemp, scratch); masm.slliw(offsetTemp, offsetTemp, 3); masm.ma_li(maskTemp, Imm32(UINT32_MAX >> ((4 - nbytes) * 8))); masm.sllw(maskTemp, maskTemp, offsetTemp); masm.nor(maskTemp, zero, maskTemp); switch (nbytes) { case 1: masm.andi(valueTemp, value, 0xff); break; case 2: masm.ma_and(valueTemp, value, Imm32(0xffff)); break; } masm.sllw(valueTemp, valueTemp, offsetTemp); masm.memoryBarrierBefore(sync); masm.bind(&again); BlockTrampolinePoolScope block_trampoline_pool(&masm, 10); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, output, scratch); masm.and_(scratch2, output, maskTemp); masm.or_(scratch2, scratch2, valueTemp); masm.sc_w(true, true, scratch2, scratch, scratch2); masm.ma_b(scratch2, Register(scratch2), &again, Assembler::NonZero, ShortJump); masm.srlw(output, output, offsetTemp); switch (nbytes) { case 1: if (signExtend) { masm.slliw(output, output, 32 - 8); masm.sraiw(output, output, 32 - 8); } else { masm.andi(valueTemp, value, 0xff); } break; case 2: if (signExtend) { masm.slliw(output, output, 32 - 16); masm.sraiw(output, output, 32 - 16); } else { masm.ma_and(valueTemp, value, Imm32(0xffff)); } break; } masm.memoryBarrierAfter(sync); } template static void AtomicExchange64(MacroAssembler& masm, const wasm::MemoryAccessDesc* access, Synchronization sync, const T& mem, Register64 value, Register64 output) { MOZ_ASSERT(value != output); UseScratchRegisterScope temps(&masm); Register SecondScratchReg = temps.Acquire(); masm.computeEffectiveAddress(mem, SecondScratchReg); Label tryAgain; masm.memoryBarrierBefore(sync); masm.bind(&tryAgain); BlockTrampolinePoolScope block_trampoline_pool(&masm, 5); if (access) { masm.append(*access, js::wasm::TrapMachineInsn::Load64, FaultingCodeOffset(masm.currentOffset())); } masm.lr_d(true, true, output.reg, SecondScratchReg); masm.movePtr(value.reg, ScratchRegister); masm.sc_d(true, true, ScratchRegister, SecondScratchReg, ScratchRegister); masm.ma_b(ScratchRegister, ScratchRegister, &tryAgain, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); } template static void AtomicFetchOp64(MacroAssembler& masm, const wasm::MemoryAccessDesc* access, Synchronization sync, AtomicOp op, Register64 value, const T& mem, Register64 temp, Register64 output) { MOZ_ASSERT(value != output); MOZ_ASSERT(value != temp); UseScratchRegisterScope temps(&masm); Register SecondScratchReg = temps.Acquire(); masm.computeEffectiveAddress(mem, SecondScratchReg); Label tryAgain; masm.memoryBarrierBefore(sync); masm.bind(&tryAgain); BlockTrampolinePoolScope block_trampoline_pool(&masm, 5); if (access) { masm.append(*access, js::wasm::TrapMachineInsn::Load64, FaultingCodeOffset(masm.currentOffset())); } masm.lr_d(true, true, output.reg, SecondScratchReg); switch (op) { case AtomicOp::Add: masm.add(temp.reg, output.reg, value.reg); break; case AtomicOp::Sub: masm.sub(temp.reg, output.reg, value.reg); break; case AtomicOp::And: masm.and_(temp.reg, output.reg, value.reg); break; case AtomicOp::Or: masm.or_(temp.reg, output.reg, value.reg); break; case AtomicOp::Xor: masm.xor_(temp.reg, output.reg, value.reg); break; default: MOZ_CRASH(); } masm.sc_d(true, true, temp.reg, SecondScratchReg, temp.reg); masm.ma_b(temp.reg, temp.reg, &tryAgain, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); } template static void AtomicEffectOp(MacroAssembler& masm, const wasm::MemoryAccessDesc* access, Scalar::Type type, Synchronization sync, AtomicOp op, const T& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp) { ScratchRegisterScope scratch(masm); UseScratchRegisterScope temps(&masm); Register scratch2 = temps.Acquire(); unsigned nbytes = Scalar::byteSize(type); switch (nbytes) { case 1: case 2: break; case 4: MOZ_ASSERT(valueTemp == InvalidReg); MOZ_ASSERT(offsetTemp == InvalidReg); MOZ_ASSERT(maskTemp == InvalidReg); break; default: MOZ_CRASH(); } Label again; masm.computeEffectiveAddress(mem, scratch); if (nbytes == 4) { masm.memoryBarrierBefore(sync); masm.bind(&again); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, scratch2, scratch); switch (op) { case AtomicOp::Add: masm.addw(scratch2, scratch2, value); break; case AtomicOp::Sub: masm.subw(scratch2, scratch2, value); break; case AtomicOp::And: masm.and_(scratch2, scratch2, value); break; case AtomicOp::Or: masm.or_(scratch2, scratch2, value); break; case AtomicOp::Xor: masm.xor_(scratch2, scratch2, value); break; default: MOZ_CRASH(); } masm.sc_w(true, true, scratch2, scratch, scratch2); masm.ma_b(scratch2, Register(scratch2), &again, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); return; } masm.andi(offsetTemp, scratch, 3); masm.subPtr(offsetTemp, scratch); masm.slliw(offsetTemp, offsetTemp, 3); masm.ma_li(maskTemp, Imm32(UINT32_MAX >> ((4 - nbytes) * 8))); masm.sllw(maskTemp, maskTemp, offsetTemp); masm.nor(maskTemp, zero, maskTemp); masm.memoryBarrierBefore(sync); masm.bind(&again); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, scratch2, scratch); masm.srlw(valueTemp, scratch2, offsetTemp); switch (op) { case AtomicOp::Add: masm.addw(valueTemp, valueTemp, value); break; case AtomicOp::Sub: masm.subw(valueTemp, valueTemp, value); break; case AtomicOp::And: masm.and_(valueTemp, valueTemp, value); break; case AtomicOp::Or: masm.or_(valueTemp, valueTemp, value); break; case AtomicOp::Xor: masm.xor_(valueTemp, valueTemp, value); break; default: MOZ_CRASH(); } switch (nbytes) { case 1: masm.andi(valueTemp, valueTemp, 0xff); break; case 2: masm.ma_and(valueTemp, valueTemp, Imm32(0xffff)); break; } masm.sllw(valueTemp, valueTemp, offsetTemp); masm.and_(scratch2, scratch2, maskTemp); masm.or_(scratch2, scratch2, valueTemp); masm.sc_w(true, true, scratch2, scratch, scratch2); masm.ma_b(scratch2, Register(scratch2), &again, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); } template static void AtomicFetchOp(MacroAssembler& masm, const wasm::MemoryAccessDesc* access, Scalar::Type type, Synchronization sync, AtomicOp op, const T& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { ScratchRegisterScope scratch(masm); UseScratchRegisterScope temps(&masm); Register scratch2 = temps.Acquire(); bool signExtend = Scalar::isSignedIntType(type); unsigned nbytes = Scalar::byteSize(type); switch (nbytes) { case 1: case 2: break; case 4: MOZ_ASSERT(valueTemp == InvalidReg); MOZ_ASSERT(offsetTemp == InvalidReg); MOZ_ASSERT(maskTemp == InvalidReg); break; default: MOZ_CRASH(); } Label again; masm.computeEffectiveAddress(mem, scratch); if (nbytes == 4) { masm.memoryBarrierBefore(sync); masm.bind(&again); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, output, scratch); switch (op) { case AtomicOp::Add: masm.addw(scratch2, output, value); break; case AtomicOp::Sub: masm.subw(scratch2, output, value); break; case AtomicOp::And: masm.and_(scratch2, output, value); break; case AtomicOp::Or: masm.or_(scratch2, output, value); break; case AtomicOp::Xor: masm.xor_(scratch2, output, value); break; default: MOZ_CRASH(); } masm.sc_w(true, true, scratch2, scratch, scratch2); masm.ma_b(scratch2, Register(scratch2), &again, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); return; } masm.andi(offsetTemp, scratch, 3); masm.subPtr(offsetTemp, scratch); masm.slliw(offsetTemp, offsetTemp, 3); masm.ma_li(maskTemp, Imm32(UINT32_MAX >> ((4 - nbytes) * 8))); masm.sllw(maskTemp, maskTemp, offsetTemp); masm.nor(maskTemp, zero, maskTemp); masm.memoryBarrierBefore(sync); masm.bind(&again); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, scratch2, scratch); masm.srlw(output, scratch2, offsetTemp); switch (op) { case AtomicOp::Add: masm.addw(valueTemp, output, value); break; case AtomicOp::Sub: masm.subw(valueTemp, output, value); break; case AtomicOp::And: masm.and_(valueTemp, output, value); break; case AtomicOp::Or: masm.or_(valueTemp, output, value); break; case AtomicOp::Xor: masm.xor_(valueTemp, output, value); break; default: MOZ_CRASH(); } switch (nbytes) { case 1: masm.andi(valueTemp, valueTemp, 0xff); break; case 2: masm.andi(valueTemp, valueTemp, 0xffff); break; } masm.sllw(valueTemp, valueTemp, offsetTemp); masm.and_(scratch2, scratch2, maskTemp); masm.or_(scratch2, scratch2, valueTemp); masm.sc_w(true, true, scratch2, scratch, scratch2); masm.ma_b(scratch2, Register(scratch2), &again, Assembler::NonZero, ShortJump); switch (nbytes) { case 1: if (signExtend) { masm.slliw(output, output, 32 - 8); masm.sraiw(output, output, 32 - 8); } else { masm.andi(output, output, 0xff); } break; case 2: if (signExtend) { masm.slliw(output, output, 32 - 16); masm.sraiw(output, output, 32 - 16); } else { masm.andi(output, output, 0xffff); } break; } masm.memoryBarrierAfter(sync); } // ======================================================================== // JS atomic operations. template static void CompareExchangeJS(MacroAssembler& masm, Scalar::Type arrayType, Synchronization sync, const T& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { if (arrayType == Scalar::Uint32) { masm.compareExchange(arrayType, sync, mem, oldval, newval, valueTemp, offsetTemp, maskTemp, temp); masm.convertUInt32ToDouble(temp, output.fpu()); } else { masm.compareExchange(arrayType, sync, mem, oldval, newval, valueTemp, offsetTemp, maskTemp, output.gpr()); } } template static void AtomicExchangeJS(MacroAssembler& masm, Scalar::Type arrayType, Synchronization sync, const T& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { if (arrayType == Scalar::Uint32) { masm.atomicExchange(arrayType, sync, mem, value, valueTemp, offsetTemp, maskTemp, temp); masm.convertUInt32ToDouble(temp, output.fpu()); } else { masm.atomicExchange(arrayType, sync, mem, value, valueTemp, offsetTemp, maskTemp, output.gpr()); } } template static void AtomicFetchOpJS(MacroAssembler& masm, Scalar::Type arrayType, Synchronization sync, AtomicOp op, Register value, const T& mem, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { if (arrayType == Scalar::Uint32) { masm.atomicFetchOp(arrayType, sync, op, value, mem, valueTemp, offsetTemp, maskTemp, temp); masm.convertUInt32ToDouble(temp, output.fpu()); } else { masm.atomicFetchOp(arrayType, sync, op, value, mem, valueTemp, offsetTemp, maskTemp, output.gpr()); } } void MacroAssembler::atomicEffectOpJS(Scalar::Type arrayType, Synchronization sync, AtomicOp op, Register value, const BaseIndex& mem, Register valueTemp, Register offsetTemp, Register maskTemp) { AtomicEffectOp(*this, nullptr, arrayType, sync, op, mem, value, valueTemp, offsetTemp, maskTemp); } void MacroAssembler::atomicEffectOpJS(Scalar::Type arrayType, Synchronization sync, AtomicOp op, Register value, const Address& mem, Register valueTemp, Register offsetTemp, Register maskTemp) { AtomicEffectOp(*this, nullptr, arrayType, sync, op, mem, value, valueTemp, offsetTemp, maskTemp); } void MacroAssembler::atomicExchange64(Synchronization sync, const Address& mem, Register64 value, Register64 output) { AtomicExchange64(*this, nullptr, sync, mem, value, output); } void MacroAssembler::atomicExchange64(Synchronization sync, const BaseIndex& mem, Register64 value, Register64 output) { AtomicExchange64(*this, nullptr, sync, mem, value, output); } void MacroAssembler::atomicExchangeJS(Scalar::Type arrayType, Synchronization sync, const Address& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { AtomicExchangeJS(*this, arrayType, sync, mem, value, valueTemp, offsetTemp, maskTemp, temp, output); } void MacroAssembler::atomicExchangeJS(Scalar::Type arrayType, Synchronization sync, const BaseIndex& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { AtomicExchangeJS(*this, arrayType, sync, mem, value, valueTemp, offsetTemp, maskTemp, temp, output); } void MacroAssembler::atomicExchange(Scalar::Type type, Synchronization sync, const Address& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicExchange(*this, nullptr, type, sync, mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::atomicExchange(Scalar::Type type, Synchronization sync, const BaseIndex& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicExchange(*this, nullptr, type, sync, mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::atomicFetchOpJS(Scalar::Type arrayType, Synchronization sync, AtomicOp op, Register value, const Address& mem, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { AtomicFetchOpJS(*this, arrayType, sync, op, value, mem, valueTemp, offsetTemp, maskTemp, temp, output); } void MacroAssembler::atomicFetchOpJS(Scalar::Type arrayType, Synchronization sync, AtomicOp op, Register value, const BaseIndex& mem, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { AtomicFetchOpJS(*this, arrayType, sync, op, value, mem, valueTemp, offsetTemp, maskTemp, temp, output); } void MacroAssembler::atomicFetchOp(Scalar::Type type, Synchronization sync, AtomicOp op, Register value, const Address& mem, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicFetchOp(*this, nullptr, type, sync, op, mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::atomicFetchOp(Scalar::Type type, Synchronization sync, AtomicOp op, Register value, const BaseIndex& mem, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicFetchOp(*this, nullptr, type, sync, op, mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::branchPtrInNurseryChunk(Condition cond, Register ptr, Register temp, Label* label) { MOZ_ASSERT(cond == Assembler::Equal || cond == Assembler::NotEqual); MOZ_ASSERT(ptr != temp); MOZ_ASSERT(ptr != ScratchRegister); // Both may be used internally. MOZ_ASSERT(temp != ScratchRegister); MOZ_ASSERT(temp != InvalidReg); ma_and(temp, ptr, Imm32(int32_t(~gc::ChunkMask))); branchPtr(InvertCondition(cond), Address(temp, gc::ChunkStoreBufferOffset), zero, label); } void MacroAssembler::branchTestValue(Condition cond, const ValueOperand& lhs, const Value& rhs, Label* label) { MOZ_ASSERT(cond == Equal || cond == NotEqual); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(lhs.valueReg() != scratch); moveValue(rhs, ValueOperand(scratch)); ma_b(lhs.valueReg(), scratch, label, cond); } void MacroAssembler::branchValueIsNurseryCell(Condition cond, const Address& address, Register temp, Label* label) { branchValueIsNurseryCellImpl(cond, address, temp, label); } void MacroAssembler::branchValueIsNurseryCell(Condition cond, ValueOperand value, Register temp, Label* label) { branchValueIsNurseryCellImpl(cond, value, temp, label); } void MacroAssembler::call(const Address& addr) { UseScratchRegisterScope temps(this); temps.Exclude(GeneralRegisterSet(1 << CallReg.code())); loadPtr(addr, CallReg); call(CallReg); } void MacroAssembler::call(ImmPtr target) { BufferOffset bo = m_buffer.nextOffset(); addPendingJump(bo, target, RelocationKind::HARDCODED); ma_call(target); } void MacroAssembler::call(ImmWord target) { call(ImmPtr((void*)target.value)); } void MacroAssembler::call(JitCode* c) { DEBUG_PRINTF("[ %s\n", __FUNCTION__); BlockTrampolinePoolScope block_trampoline_pool(this, 8); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BufferOffset bo = m_buffer.nextOffset(); addPendingJump(bo, ImmPtr(c->raw()), RelocationKind::JITCODE); ma_liPatchable(scratch, ImmPtr(c->raw())); callJitNoProfiler(scratch); DEBUG_PRINTF("]\n"); } void MacroAssembler::callWithABIPre(uint32_t* stackAdjust, bool callFromWasm) { MOZ_ASSERT(inCall_); uint32_t stackForCall = abiArgs_.stackBytesConsumedSoFar(); // Reserve place for $ra. stackForCall += sizeof(intptr_t); if (dynamicAlignment_) { stackForCall += ComputeByteAlignment(stackForCall, ABIStackAlignment); } else { uint32_t alignmentAtPrologue = callFromWasm ? sizeof(wasm::Frame) : 0; stackForCall += ComputeByteAlignment( stackForCall + framePushed() + alignmentAtPrologue, ABIStackAlignment); } *stackAdjust = stackForCall; reserveStack(stackForCall); // Save $ra because call is going to clobber it. Restore it in // callWithABIPost. NOTE: This is needed for calls from SharedIC. // Maybe we can do this differently. storePtr(ra, Address(StackPointer, stackForCall - sizeof(intptr_t))); // Position all arguments. { enoughMemory_ &= moveResolver_.resolve(); if (!enoughMemory_) { return; } MoveEmitter emitter(asMasm()); emitter.emit(moveResolver_); emitter.finish(); } assertStackAlignment(ABIStackAlignment); } void MacroAssembler::callWithABIPost(uint32_t stackAdjust, ABIType result, bool callFromWasm) { // Restore ra value (as stored in callWithABIPre()). loadPtr(Address(StackPointer, stackAdjust - sizeof(intptr_t)), ra); if (dynamicAlignment_) { // Restore sp value from stack (as stored in setupUnalignedABICall()). loadPtr(Address(StackPointer, stackAdjust), StackPointer); // Use adjustFrame instead of freeStack because we already restored sp. adjustFrame(-stackAdjust); } else { freeStack(stackAdjust); } #ifdef DEBUG MOZ_ASSERT(inCall_); inCall_ = false; #endif } void MacroAssembler::callWithABINoProfiler(Register fun, ABIType result) { // Load the callee in scratch2, no instruction between the movePtr and // call should clobber it. Note that we can't use fun because it may be // one of the IntArg registers clobbered before the call. UseScratchRegisterScope temps(this); temps.Exclude(GeneralRegisterSet(1 << CallReg.code())); movePtr(fun, CallReg); uint32_t stackAdjust; callWithABIPre(&stackAdjust); call(CallReg); callWithABIPost(stackAdjust, result); } void MacroAssembler::callWithABINoProfiler(const Address& fun, ABIType result) { // Load the callee in scratch2, as above. UseScratchRegisterScope temps(this); temps.Exclude(GeneralRegisterSet(1 << CallReg.code())); loadPtr(fun, CallReg); uint32_t stackAdjust; callWithABIPre(&stackAdjust); call(CallReg); callWithABIPost(stackAdjust, result); } void MacroAssembler::ceilDoubleToInt32(FloatRegister src, Register dest, Label* fail) { UseScratchRegisterScope temps(this); ScratchDoubleScope fscratch(*this); Label performCeil, done; // If x < -1 or x > 0 then perform ceil. loadConstantDouble(0, fscratch); branchDouble(Assembler::DoubleGreaterThan, src, fscratch, &performCeil); loadConstantDouble(-1.0, fscratch); branchDouble(Assembler::DoubleLessThanOrEqual, src, fscratch, &performCeil); Register scratch = temps.Acquire(); // If binary value is not zero, the input was not 0, so we bail. { moveFromDoubleHi(src, scratch); branch32(Assembler::NotEqual, scratch, zero, fail); } bind(&performCeil); Ceil_w_d(dest, src, scratch); ma_b(scratch, Imm32(1), fail, NotEqual); bind(&done); } void MacroAssembler::ceilFloat32ToInt32(FloatRegister src, Register dest, Label* fail) { UseScratchRegisterScope temps(this); ScratchDoubleScope fscratch(*this); Label performCeil, done; // If x < -1 or x > 0 then perform ceil. loadConstantFloat32(0, fscratch); branchFloat(Assembler::DoubleGreaterThan, src, fscratch, &performCeil); loadConstantFloat32(-1.0, fscratch); branchFloat(Assembler::DoubleLessThanOrEqual, src, fscratch, &performCeil); Register scratch = temps.Acquire(); // If binary value is not zero, the input was not 0, so we bail. { fmv_x_w(scratch, src); branch32(Assembler::NotEqual, scratch, zero, fail); } bind(&performCeil); Ceil_w_s(dest, src, scratch); ma_b(scratch, Imm32(1), fail, NotEqual); bind(&done); } void MacroAssembler::comment(const char* msg) { Assembler::comment(msg); } template static void CompareExchange64(MacroAssembler& masm, const wasm::MemoryAccessDesc* access, Synchronization sync, const T& mem, Register64 expect, Register64 replace, Register64 output) { MOZ_ASSERT(expect != output && replace != output); ScratchRegisterScope scratch(masm); UseScratchRegisterScope temps(&masm); Register scratch2 = temps.Acquire(); masm.computeEffectiveAddress(mem, scratch); Label tryAgain; Label exit; masm.memoryBarrierBefore(sync); masm.bind(&tryAgain); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_d(true, true, output.reg, scratch); masm.ma_b(output.reg, expect.reg, &exit, Assembler::NotEqual, ShortJump); masm.movePtr(replace.reg, scratch2); masm.sc_d(true, true, scratch2, scratch, scratch2); masm.ma_b(scratch2, Register(scratch2), &tryAgain, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); masm.bind(&exit); } void MacroAssembler::compareExchange64(Synchronization sync, const Address& mem, Register64 expect, Register64 replace, Register64 output) { CompareExchange64(*this, nullptr, sync, mem, expect, replace, output); } void MacroAssembler::compareExchange64(Synchronization sync, const BaseIndex& mem, Register64 expect, Register64 replace, Register64 output) { CompareExchange64(*this, nullptr, sync, mem, expect, replace, output); } void MacroAssembler::compareExchangeJS(Scalar::Type arrayType, Synchronization sync, const Address& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { CompareExchangeJS(*this, arrayType, sync, mem, oldval, newval, valueTemp, offsetTemp, maskTemp, temp, output); } void MacroAssembler::compareExchangeJS(Scalar::Type arrayType, Synchronization sync, const BaseIndex& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register temp, AnyRegister output) { CompareExchangeJS(*this, arrayType, sync, mem, oldval, newval, valueTemp, offsetTemp, maskTemp, temp, output); } void MacroAssembler::convertInt64ToDouble(Register64 src, FloatRegister dest) { fcvt_d_l(dest, src.scratchReg()); } void MacroAssembler::convertInt64ToFloat32(Register64 src, FloatRegister dest) { fcvt_s_l(dest, src.scratchReg()); } void MacroAssembler::convertIntPtrToDouble(Register src, FloatRegister dest) { fcvt_d_l(dest, src); } void MacroAssembler::convertUInt64ToDouble(Register64 src, FloatRegister dest, Register tmp) { fcvt_d_lu(dest, src.scratchReg()); } void MacroAssembler::convertUInt64ToFloat32(Register64 src, FloatRegister dest, Register tmp) { fcvt_s_lu(dest, src.scratchReg()); } void MacroAssembler::copySignDouble(FloatRegister lhs, FloatRegister rhs, FloatRegister output) { fsgnj_d(output, lhs, rhs); } void MacroAssembler::enterFakeExitFrameForWasm(Register cxreg, Register scratch, ExitFrameType type) { enterFakeExitFrame(cxreg, scratch, type); } void MacroAssembler::flexibleDivMod32(Register rhs, Register srcDest, Register remOutput, bool isUnsigned, const LiveRegisterSet&) { if (isUnsigned) { ma_modu32(remOutput, srcDest, rhs); ma_divu32(srcDest, srcDest, rhs); } else { ma_mod32(remOutput, srcDest, rhs); ma_div32(srcDest, srcDest, rhs); } } void MacroAssembler::flexibleQuotient32(Register rhs, Register srcDest, bool isUnsigned, const LiveRegisterSet&) { quotient32(rhs, srcDest, isUnsigned); } void MacroAssembler::flexibleRemainder32(Register rhs, Register srcDest, bool isUnsigned, const LiveRegisterSet&) { remainder32(rhs, srcDest, isUnsigned); } void MacroAssembler::floorDoubleToInt32(FloatRegister src, Register dest, Label* fail) { JitSpew(JitSpew_Codegen, "[ %s", __FUNCTION__); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Floor_w_d(dest, src, scratch); ma_b(scratch, Imm32(1), fail, NotEqual); fmv_x_d(scratch, src); ma_branch(fail, Equal, scratch, Operand(0x8000000000000000)); JitSpew(JitSpew_Codegen, "]"); } void MacroAssembler::floorFloat32ToInt32(FloatRegister src, Register dest, Label* fail) { JitSpew(JitSpew_Codegen, "[ %s", __FUNCTION__); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Floor_w_s(dest, src, scratch); ma_b(scratch, Imm32(1), fail, NotEqual); fmv_x_w(scratch, src); ma_branch(fail, Equal, scratch, Operand(int32_t(0x80000000))); JitSpew(JitSpew_Codegen, "]"); } void MacroAssembler::flush() {} void MacroAssembler::loadStoreBuffer(Register ptr, Register buffer) { ma_and(buffer, ptr, Imm32(int32_t(~gc::ChunkMask))); loadPtr(Address(buffer, gc::ChunkStoreBufferOffset), buffer); } void MacroAssembler::moveValue(const TypedOrValueRegister& src, const ValueOperand& dest) { if (src.hasValue()) { moveValue(src.valueReg(), dest); return; } MIRType type = src.type(); AnyRegister reg = src.typedReg(); if (!IsFloatingPointType(type)) { boxNonDouble(ValueTypeFromMIRType(type), reg.gpr(), dest); return; } ScratchDoubleScope fpscratch(asMasm()); FloatRegister scratch = fpscratch; FloatRegister freg = reg.fpu(); if (type == MIRType::Float32) { convertFloat32ToDouble(freg, scratch); freg = scratch; } boxDouble(freg, dest, scratch); } void MacroAssembler::moveValue(const ValueOperand& src, const ValueOperand& dest) { if (src == dest) { return; } movePtr(src.valueReg(), dest.valueReg()); } void MacroAssembler::moveValue(const Value& src, const ValueOperand& dest) { if (!src.isGCThing()) { ma_li(dest.valueReg(), ImmWord(src.asRawBits())); return; } writeDataRelocation(src); movWithPatch(ImmWord(src.asRawBits()), dest.valueReg()); } void MacroAssembler::nearbyIntDouble(RoundingMode, FloatRegister, FloatRegister) { MOZ_CRASH("not supported on this platform"); } void MacroAssembler::nearbyIntFloat32(RoundingMode, FloatRegister, FloatRegister) { MOZ_CRASH("not supported on this platform"); } void MacroAssembler::oolWasmTruncateCheckF32ToI32(FloatRegister input, Register output, TruncFlags flags, wasm::BytecodeOffset off, Label* rejoin) { Label notNaN; UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); CompareIsNotNanF32(scratch, input, input); ma_branch(¬NaN, Equal, scratch, Operand(1)); wasmTrap(wasm::Trap::InvalidConversionToInteger, off); bind(¬NaN); Label isOverflow; const float two_31 = -float(INT32_MIN); ScratchFloat32Scope fpscratch(*this); if (flags & TRUNC_UNSIGNED) { loadConstantFloat32(two_31 * 2, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantFloat32(-1.0f, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThan, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } else { loadConstantFloat32(two_31, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantFloat32(-two_31, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } bind(&isOverflow); wasmTrap(wasm::Trap::IntegerOverflow, off); } void MacroAssembler::oolWasmTruncateCheckF64ToI32(FloatRegister input, Register output, TruncFlags flags, wasm::BytecodeOffset off, Label* rejoin) { Label notNaN; UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); CompareIsNotNanF64(scratch, input, input); ma_branch(¬NaN, Equal, scratch, Operand(1)); wasmTrap(wasm::Trap::InvalidConversionToInteger, off); bind(¬NaN); Label isOverflow; const double two_31 = -double(INT32_MIN); ScratchDoubleScope fpscratch(*this); if (flags & TRUNC_UNSIGNED) { loadConstantDouble(two_31 * 2, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantDouble(-1.0, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThan, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } else { loadConstantDouble(two_31, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantDouble(-two_31 - 1, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThan, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } bind(&isOverflow); wasmTrap(wasm::Trap::IntegerOverflow, off); } void MacroAssembler::oolWasmTruncateCheckF32ToI64(FloatRegister input, Register64 output, TruncFlags flags, wasm::BytecodeOffset off, Label* rejoin) { Label notNaN; UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); CompareIsNotNanF32(scratch, input, input); ma_branch(¬NaN, Equal, scratch, Operand(1)); wasmTrap(wasm::Trap::InvalidConversionToInteger, off); bind(¬NaN); Label isOverflow; const float two_63 = -float(INT64_MIN); ScratchFloat32Scope fpscratch(*this); if (flags & TRUNC_UNSIGNED) { loadConstantFloat32(two_63 * 2, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantFloat32(-1.0f, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThan, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } else { loadConstantFloat32(two_63, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantFloat32(-two_63, fpscratch); ma_compareF32(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } bind(&isOverflow); wasmTrap(wasm::Trap::IntegerOverflow, off); } void MacroAssembler::oolWasmTruncateCheckF64ToI64(FloatRegister input, Register64 output, TruncFlags flags, wasm::BytecodeOffset off, Label* rejoin) { Label notNaN; UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); CompareIsNotNanF64(scratch, input, input); ma_branch(¬NaN, Equal, scratch, Operand(1)); wasmTrap(wasm::Trap::InvalidConversionToInteger, off); bind(¬NaN); Label isOverflow; const double two_63 = -double(INT64_MIN); ScratchDoubleScope fpscratch(*this); if (flags & TRUNC_UNSIGNED) { loadConstantDouble(two_63 * 2, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantDouble(-1.0, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThan, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } else { loadConstantDouble(two_63, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThanOrEqual, input, fpscratch); ma_branch(&isOverflow, Equal, scratch, Operand(1)); loadConstantDouble(-two_63, fpscratch); ma_compareF64(scratch, Assembler::DoubleGreaterThan, input, fpscratch); ma_b(scratch, Imm32(1), rejoin, Equal); } bind(&isOverflow); wasmTrap(wasm::Trap::IntegerOverflow, off); } void MacroAssembler::patchCallToNop(uint8_t* call) { uint32_t* p = reinterpret_cast(call) - 7; *reinterpret_cast(p) = kNopByte; *reinterpret_cast(p + 1) = kNopByte; *reinterpret_cast(p + 2) = kNopByte; *reinterpret_cast(p + 3) = kNopByte; *reinterpret_cast(p + 4) = kNopByte; *reinterpret_cast(p + 5) = kNopByte; *reinterpret_cast(p + 6) = kNopByte; } CodeOffset MacroAssembler::callWithPatch() { BlockTrampolinePoolScope block_trampoline_pool(this, 2); DEBUG_PRINTF("\tcallWithPatch\n"); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); int32_t imm32 = 1 * sizeof(uint32_t); int32_t Hi20 = ((imm32 + 0x800) >> 12); int32_t Lo12 = imm32 << 20 >> 20; auipc(scratch, Hi20); // Read PC + Hi20 into scratch. jalr(scratch, Lo12); // jump PC + Hi20 + Lo12 DEBUG_PRINTF("\tret %d\n", currentOffset()); return CodeOffset(currentOffset()); } void MacroAssembler::patchCall(uint32_t callerOffset, uint32_t calleeOffset) { DEBUG_PRINTF("\tpatchCall\n"); BufferOffset call(callerOffset - 2 * sizeof(uint32_t)); DEBUG_PRINTF("\tcallerOffset %d\n", callerOffset); int32_t offset = BufferOffset(calleeOffset).getOffset() - call.getOffset(); if (is_int32(offset)) { Instruction* auipc_ = (Instruction*)editSrc(call); Instruction* jalr_ = (Instruction*)editSrc( BufferOffset(callerOffset - 1 * sizeof(uint32_t))); DEBUG_PRINTF("\t%p %lu\n\t", auipc_, callerOffset - 2 * sizeof(uint32_t)); disassembleInstr(auipc_->InstructionBits()); DEBUG_PRINTF("\t%p %lu\n\t", jalr_, callerOffset - 1 * sizeof(uint32_t)); disassembleInstr(jalr_->InstructionBits()); DEBUG_PRINTF("\t\n"); MOZ_ASSERT(IsJalr(jalr_->InstructionBits()) && IsAuipc(auipc_->InstructionBits())); MOZ_ASSERT(auipc_->RdValue() == jalr_->Rs1Value()); int32_t Hi20 = (((int32_t)offset + 0x800) >> 12); int32_t Lo12 = (int32_t)offset << 20 >> 20; instr_at_put(call, SetAuipcOffset(Hi20, auipc_->InstructionBits())); instr_at_put(BufferOffset(callerOffset - 1 * sizeof(uint32_t)), SetJalrOffset(Lo12, jalr_->InstructionBits())); } else { MOZ_CRASH(); } } void MacroAssembler::patchFarJump(CodeOffset farJump, uint32_t targetOffset) { uint32_t* u32 = reinterpret_cast( editSrc(BufferOffset(farJump.offset() + 4 * kInstrSize))); MOZ_ASSERT(*u32 == UINT32_MAX); *u32 = targetOffset - farJump.offset(); } void MacroAssembler::patchNearAddressMove(CodeLocationLabel loc, CodeLocationLabel target) { PatchDataWithValueCheck(loc, ImmPtr(target.raw()), ImmPtr(nullptr)); } void MacroAssembler::patchNopToCall(uint8_t* call, uint8_t* target) { uint32_t* p = reinterpret_cast(call) - 7; Assembler::WriteLoad64Instructions((Instruction*)p, ScratchRegister, (uint64_t)target); DEBUG_PRINTF("\tpatchNopToCall %lu %lu\n", (uint64_t)target, ExtractLoad64Value((Instruction*)p)); MOZ_ASSERT(ExtractLoad64Value((Instruction*)p) == (uint64_t)target); Instr jalr_ = JALR | (ra.code() << kRdShift) | (0x0 << kFunct3Shift) | (ScratchRegister.code() << kRs1Shift) | (0x0 << kImm12Shift); *reinterpret_cast(p + 6) = jalr_; } void MacroAssembler::Pop(Register reg) { ma_pop(reg); adjustFrame(-int32_t(sizeof(intptr_t))); } void MacroAssembler::Pop(FloatRegister f) { ma_pop(f); adjustFrame(-int32_t(sizeof(double))); } void MacroAssembler::Pop(const ValueOperand& val) { popValue(val); adjustFrame(-int32_t(sizeof(Value))); } void MacroAssembler::PopRegsInMaskIgnore(LiveRegisterSet set, LiveRegisterSet ignore) { int32_t diff = set.gprs().size() * sizeof(intptr_t) + set.fpus().getPushSizeInBytes(); const int32_t reserved = diff; for (GeneralRegisterBackwardIterator iter(set.gprs()); iter.more(); ++iter) { diff -= sizeof(intptr_t); if (!ignore.has(*iter)) { loadPtr(Address(StackPointer, diff), *iter); } } #ifdef ENABLE_WASM_SIMD # error "Needs more careful logic if SIMD is enabled" #endif for (FloatRegisterBackwardIterator iter(set.fpus().reduceSetForPush()); iter.more(); ++iter) { diff -= sizeof(double); if (!ignore.has(*iter)) { loadDouble(Address(StackPointer, diff), *iter); } } MOZ_ASSERT(diff == 0); freeStack(reserved); } void MacroAssembler::pushReturnAddress() { push(ra); } void MacroAssembler::popReturnAddress() { pop(ra); } void MacroAssembler::PopStackPtr() { loadPtr(Address(StackPointer, 0), StackPointer); adjustFrame(-int32_t(sizeof(intptr_t))); } void MacroAssembler::freeStackTo(uint32_t framePushed) { MOZ_ASSERT(framePushed <= framePushed_); ma_sub64(StackPointer, FramePointer, Imm32(framePushed)); framePushed_ = framePushed; } void MacroAssembler::PushBoxed(FloatRegister reg) { subFromStackPtr(Imm32(sizeof(double))); boxDouble(reg, Address(getStackPointer(), 0)); adjustFrame(sizeof(double)); } void MacroAssembler::Push(Register reg) { ma_push(reg); adjustFrame(int32_t(sizeof(intptr_t))); } void MacroAssembler::Push(const Imm32 imm) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, imm); ma_push(scratch); adjustFrame(int32_t(sizeof(intptr_t))); } void MacroAssembler::Push(const ImmWord imm) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, imm); ma_push(scratch); adjustFrame(int32_t(sizeof(intptr_t))); } void MacroAssembler::Push(const ImmPtr imm) { Push(ImmWord(uintptr_t(imm.value))); } void MacroAssembler::Push(const ImmGCPtr ptr) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, ptr); ma_push(scratch); adjustFrame(int32_t(sizeof(intptr_t))); } void MacroAssembler::Push(FloatRegister f) { ma_push(f); adjustFrame(int32_t(sizeof(double))); } void MacroAssembler::PushRegsInMask(LiveRegisterSet set) { int32_t diff = set.gprs().size() * sizeof(intptr_t) + set.fpus().getPushSizeInBytes(); const int32_t reserved = diff; reserveStack(reserved); for (GeneralRegisterBackwardIterator iter(set.gprs()); iter.more(); ++iter) { diff -= sizeof(intptr_t); storePtr(*iter, Address(StackPointer, diff)); } #ifdef ENABLE_WASM_SIMD # error "Needs more careful logic if SIMD is enabled" #endif for (FloatRegisterBackwardIterator iter(set.fpus().reduceSetForPush()); iter.more(); ++iter) { diff -= sizeof(double); storeDouble(*iter, Address(StackPointer, diff)); } MOZ_ASSERT(diff == 0); } void MacroAssembler::roundFloat32ToInt32(FloatRegister src, Register dest, FloatRegister temp, Label* fail) { JitSpew(JitSpew_Codegen, "[ %s", __FUNCTION__); ScratchDoubleScope fscratch(*this); Label negative, done; // Branch to a slow path if input < 0.0 due to complicated rounding rules. // Note that Fcmp with NaN unsets the negative flag. { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); fmv_x_w(scratch, src); ma_branch(fail, Equal, scratch, Operand(int32_t(0x80000000))); fmv_w_x(temp, zero); ma_compareF32(scratch, DoubleLessThan, src, temp); ma_branch(&negative, Equal, scratch, Operand(1)); } // Handle the simple case of a positive input, and also -0 and NaN. // Rounding proceeds with consideration of the fractional part of the input: // 1. If > 0.5, round to integer with higher absolute value (so, up). // 2. If < 0.5, round to integer with lower absolute value (so, down). // 3. If = 0.5, round to +Infinity (so, up). { // Convert to signed 32-bit integer, rounding halfway cases away from zero. // In the case of overflow, the output is saturated. // In the case of NaN and -0, the output is zero. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); RoundFloatingPointToInteger( dest, src, scratch, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_s(dst, src, RMM); }, false); ma_b(scratch, Imm32(1), fail, NotEqual); jump(&done); } // Handle the complicated case of a negative input. // Rounding proceeds with consideration of the fractional part of the input: // 1. If > 0.5, round to integer with higher absolute value (so, down). // 2. If < 0.5, round to integer with lower absolute value (so, up). // 3. If = 0.5, round to +Infinity (so, up). bind(&negative); { // Inputs in [-0.5, 0) need 0.5 added; other negative inputs need // the biggest double less than 0.5. Label join; loadConstantFloat32(GetBiggestNumberLessThan(0.5), temp); loadConstantFloat32(-0.5, fscratch); branchFloat(Assembler::DoubleLessThan, src, fscratch, &join); loadConstantFloat32(0.5, temp); bind(&join); addFloat32(src, temp); // Round all values toward -Infinity. // In the case of overflow, the output is saturated. // NaN and -0 are already handled by the "positive number" path above. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); RoundFloatingPointToInteger( dest, temp, scratch, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_s(dst, src, RDN); }, false); ma_b(scratch, Imm32(1), fail, NotEqual); // If output is zero, then the actual result is -0. Fail. branchTest32(Assembler::Zero, dest, dest, fail); } bind(&done); JitSpew(JitSpew_Codegen, "]"); } void MacroAssembler::roundDoubleToInt32(FloatRegister src, Register dest, FloatRegister temp, Label* fail) { JitSpew(JitSpew_Codegen, "[ %s", __FUNCTION__); ScratchDoubleScope fscratch(*this); Label negative, done; // Branch to a slow path if input < 0.0 due to complicated rounding rules. // Note that Fcmp with NaN unsets the negative flag. { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); fmv_x_d(scratch, src); ma_branch(fail, Equal, scratch, Operand(0x8000000000000000)); fmv_d_x(temp, zero); ma_compareF64(scratch, DoubleLessThan, src, temp); ma_branch(&negative, Equal, scratch, Operand(1)); } // Handle the simple case of a positive input, and also -0 and NaN. // Rounding proceeds with consideration of the fractional part of the input: // 1. If > 0.5, round to integer with higher absolute value (so, up). // 2. If < 0.5, round to integer with lower absolute value (so, down). // 3. If = 0.5, round to +Infinity (so, up). { // Convert to signed 32-bit integer, rounding halfway cases away from zero. // In the case of overflow, the output is saturated. // In the case of NaN and -0, the output is zero. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); RoundFloatingPointToInteger( dest, src, scratch, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_d(dst, src, RMM); }, false); ma_b(scratch, Imm32(1), fail, NotEqual); jump(&done); } // Handle the complicated case of a negative input. // Rounding proceeds with consideration of the fractional part of the input: // 1. If > 0.5, round to integer with higher absolute value (so, down). // 2. If < 0.5, round to integer with lower absolute value (so, up). // 3. If = 0.5, round to +Infinity (so, up). bind(&negative); { // Inputs in [-0.5, 0) need 0.5 added; other negative inputs need // the biggest double less than 0.5. Label join; loadConstantDouble(GetBiggestNumberLessThan(0.5), temp); loadConstantDouble(-0.5, fscratch); branchDouble(Assembler::DoubleLessThan, src, fscratch, &join); loadConstantDouble(0.5, temp); bind(&join); addDouble(src, temp); // Round all values toward -Infinity. // In the case of overflow, the output is saturated. // NaN and -0 are already handled by the "positive number" path above. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); RoundFloatingPointToInteger( dest, temp, scratch, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_d(dst, src, RDN); }, false); ma_b(scratch, Imm32(1), fail, NotEqual); // If output is zero, then the actual result is -0. Fail. branchTest32(Assembler::Zero, dest, dest, fail); } bind(&done); JitSpew(JitSpew_Codegen, "]"); } void MacroAssembler::setupUnalignedABICall(Register scratch) { MOZ_ASSERT(!IsCompilingWasm(), "wasm should only use aligned ABI calls"); setupNativeABICall(); dynamicAlignment_ = true; or_(scratch, StackPointer, zero); // Force sp to be aligned asMasm().subPtr(Imm32(sizeof(uintptr_t)), StackPointer); ma_and(StackPointer, StackPointer, Imm32(~(ABIStackAlignment - 1))); storePtr(scratch, Address(StackPointer, 0)); } void MacroAssembler::shiftIndex32AndAdd(Register indexTemp32, int shift, Register pointer) { if (IsShiftInScaleRange(shift)) { computeEffectiveAddress( BaseIndex(pointer, indexTemp32, ShiftToScale(shift)), pointer); return; } lshift32(Imm32(shift), indexTemp32); addPtr(indexTemp32, pointer); } void MacroAssembler::speculationBarrier() { MOZ_CRASH(); } void MacroAssembler::storeRegsInMask(LiveRegisterSet set, Address dest, Register) { FloatRegisterSet fpuSet(set.fpus().reduceSetForPush()); unsigned numFpu = fpuSet.size(); int32_t diffF = fpuSet.getPushSizeInBytes(); int32_t diffG = set.gprs().size() * sizeof(intptr_t); MOZ_ASSERT(dest.offset >= diffG + diffF); for (GeneralRegisterBackwardIterator iter(set.gprs()); iter.more(); ++iter) { diffG -= sizeof(intptr_t); dest.offset -= sizeof(intptr_t); storePtr(*iter, dest); } MOZ_ASSERT(diffG == 0); #ifdef ENABLE_WASM_SIMD # error "Needs more careful logic if SIMD is enabled" #endif for (FloatRegisterBackwardIterator iter(fpuSet); iter.more(); ++iter) { FloatRegister reg = *iter; diffF -= reg.size(); numFpu -= 1; dest.offset -= reg.size(); if (reg.isDouble()) { storeDouble(reg, dest); } else if (reg.isSingle()) { storeFloat32(reg, dest); } else { MOZ_CRASH("Unknown register type."); } } MOZ_ASSERT(numFpu == 0); diffF -= diffF % sizeof(uintptr_t); MOZ_ASSERT(diffF == 0); } void MacroAssembler::truncDoubleToInt32(FloatRegister src, Register dest, Label* fail) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Label zeroCase, done; // Convert scalar to signed 32-bit fixed-point, rounding toward zero. // In the case of overflow, the output is saturated. // In the case of NaN and -0, the output is zero. RoundFloatingPointToInteger( dest, src, scratch, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_d(dst, src, RTZ); }, false); ma_b(scratch, Imm32(1), fail, NotEqual); // If the output was zero, worry about special cases. branch32(Assembler::Equal, dest, Imm32(0), &zeroCase); jump(&done); // Handle the case of a zero output: // 1. The input may have been NaN, requiring a failure. // 2. The input may have been in (-1,-0], requiring a failure. // 3. +0, return 0. { bind(&zeroCase); // If input is a negative number that truncated to zero, the real // output should be the non-integer -0. // The use of "lt" instead of "lo" also catches unordered NaN input. ScratchDoubleScope fscratch(*this); fmv_d_x(fscratch, zero); ma_compareF64(scratch, DoubleLessThan, src, fscratch); ma_b(scratch, Imm32(1), fail, Equal); // Check explicitly for -0, bitwise. fmv_x_d(dest, src); branchTestPtr(Assembler::Signed, dest, dest, fail); movePtr(ImmWord(0), dest); } bind(&done); } void MacroAssembler::truncFloat32ToInt32(FloatRegister src, Register dest, Label* fail) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Label zeroCase, done; // Convert scalar to signed 32-bit fixed-point, rounding toward zero. // In the case of overflow, the output is saturated. // In the case of NaN and -0, the output is zero. RoundFloatingPointToInteger( dest, src, scratch, [](MacroAssemblerRiscv64* masm, Register dst, FPURegister src) { masm->fcvt_w_s(dst, src, RTZ); }, false); ma_b(scratch, Imm32(1), fail, NotEqual); // If the output was zero, worry about special cases. branch32(Assembler::Equal, dest, Imm32(0), &zeroCase); jump(&done); // Handle the case of a zero output: // 1. The input may have been NaN, requiring a failure. // 2. The input may have been in (-1,-0], requiring a failure. // 3. +0, return 0. { bind(&zeroCase); // If input is a negative number that truncated to zero, the real // output should be the non-integer -0. // The use of "lt" instead of "lo" also catches unordered NaN input. ScratchDoubleScope fscratch(*this); fmv_w_x(fscratch, zero); ma_compareF32(scratch, DoubleLessThan, src, fscratch); ma_b(scratch, Imm32(1), fail, Equal); // Check explicitly for -0, bitwise. fmv_x_w(dest, src); branchTestPtr(Assembler::Signed, dest, dest, fail); movePtr(ImmWord(0), dest); } bind(&done); } void MacroAssembler::wasmAtomicEffectOp(const wasm::MemoryAccessDesc& access, AtomicOp op, Register value, const Address& mem, Register valueTemp, Register offsetTemp, Register maskTemp) { AtomicEffectOp(*this, &access, access.type(), access.sync(), op, mem, value, valueTemp, offsetTemp, maskTemp); } void MacroAssembler::wasmAtomicEffectOp(const wasm::MemoryAccessDesc& access, AtomicOp op, Register value, const BaseIndex& mem, Register valueTemp, Register offsetTemp, Register maskTemp) { AtomicEffectOp(*this, &access, access.type(), access.sync(), op, mem, value, valueTemp, offsetTemp, maskTemp); } template static void WasmAtomicExchange64(MacroAssembler& masm, const wasm::MemoryAccessDesc& access, const T& mem, Register64 value, Register64 output) { AtomicExchange64(masm, &access, access.sync(), mem, value, output); } void MacroAssembler::wasmAtomicExchange64(const wasm::MemoryAccessDesc& access, const Address& mem, Register64 src, Register64 output) { WasmAtomicExchange64(*this, access, mem, src, output); } void MacroAssembler::wasmAtomicExchange64(const wasm::MemoryAccessDesc& access, const BaseIndex& mem, Register64 src, Register64 output) { WasmAtomicExchange64(*this, access, mem, src, output); } void MacroAssembler::wasmAtomicExchange(const wasm::MemoryAccessDesc& access, const Address& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicExchange(*this, &access, access.type(), access.sync(), mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::wasmAtomicExchange(const wasm::MemoryAccessDesc& access, const BaseIndex& mem, Register value, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicExchange(*this, &access, access.type(), access.sync(), mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::wasmAtomicFetchOp64(const wasm::MemoryAccessDesc& access, AtomicOp op, Register64 value, const Address& mem, Register64 temp, Register64 output) { AtomicFetchOp64(*this, &access, access.sync(), op, value, mem, temp, output); } void MacroAssembler::wasmAtomicFetchOp64(const wasm::MemoryAccessDesc& access, AtomicOp op, Register64 value, const BaseIndex& mem, Register64 temp, Register64 output) { AtomicFetchOp64(*this, &access, access.sync(), op, value, mem, temp, output); } void MacroAssembler::atomicFetchOp64(Synchronization sync, AtomicOp op, Register64 value, const Address& mem, Register64 temp, Register64 output) { AtomicFetchOp64(*this, nullptr, sync, op, value, mem, temp, output); } void MacroAssembler::atomicFetchOp64(Synchronization sync, AtomicOp op, Register64 value, const BaseIndex& mem, Register64 temp, Register64 output) { AtomicFetchOp64(*this, nullptr, sync, op, value, mem, temp, output); } void MacroAssembler::atomicEffectOp64(Synchronization sync, AtomicOp op, Register64 value, const Address& mem, Register64 temp) { AtomicFetchOp64(*this, nullptr, sync, op, value, mem, temp, temp); } void MacroAssembler::atomicEffectOp64(Synchronization sync, AtomicOp op, Register64 value, const BaseIndex& mem, Register64 temp) { AtomicFetchOp64(*this, nullptr, sync, op, value, mem, temp, temp); } void MacroAssembler::wasmAtomicFetchOp(const wasm::MemoryAccessDesc& access, AtomicOp op, Register value, const Address& mem, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicFetchOp(*this, &access, access.type(), access.sync(), op, mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::wasmAtomicFetchOp(const wasm::MemoryAccessDesc& access, AtomicOp op, Register value, const BaseIndex& mem, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { AtomicFetchOp(*this, &access, access.type(), access.sync(), op, mem, value, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::wasmBoundsCheck32(Condition cond, Register index, Register boundsCheckLimit, Label* ok) { ma_b(index, boundsCheckLimit, ok, cond); } void MacroAssembler::wasmBoundsCheck32(Condition cond, Register index, Address boundsCheckLimit, Label* ok) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); load32(boundsCheckLimit, scratch2); ma_b(index, Register(scratch2), ok, cond); } void MacroAssembler::wasmBoundsCheck64(Condition cond, Register64 index, Register64 boundsCheckLimit, Label* ok) { ma_b(index.reg, boundsCheckLimit.reg, ok, cond); } void MacroAssembler::wasmBoundsCheck64(Condition cond, Register64 index, Address boundsCheckLimit, Label* ok) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); loadPtr(boundsCheckLimit, scratch2); ma_b(index.reg, scratch2, ok, cond); } void MacroAssembler::wasmCompareExchange64(const wasm::MemoryAccessDesc& access, const Address& mem, Register64 expect, Register64 replace, Register64 output) { CompareExchange64(*this, &access, access.sync(), mem, expect, replace, output); } void MacroAssembler::wasmCompareExchange64(const wasm::MemoryAccessDesc& access, const BaseIndex& mem, Register64 expect, Register64 replace, Register64 output) { CompareExchange64(*this, &access, access.sync(), mem, expect, replace, output); } template static void CompareExchange(MacroAssembler& masm, const wasm::MemoryAccessDesc* access, Scalar::Type type, Synchronization sync, const T& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { bool signExtend = Scalar::isSignedIntType(type); unsigned nbytes = Scalar::byteSize(type); switch (nbytes) { case 1: case 2: break; case 4: MOZ_ASSERT(valueTemp == InvalidReg); MOZ_ASSERT(offsetTemp == InvalidReg); MOZ_ASSERT(maskTemp == InvalidReg); break; default: MOZ_CRASH(); } Label again, end; UseScratchRegisterScope temps(&masm); Register SecondScratchReg = temps.Acquire(); masm.computeEffectiveAddress(mem, SecondScratchReg); if (nbytes == 4) { masm.memoryBarrierBefore(sync); masm.bind(&again); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, output, SecondScratchReg); masm.ma_b(output, oldval, &end, Assembler::NotEqual, ShortJump); masm.mv(ScratchRegister, newval); masm.sc_w(true, true, ScratchRegister, SecondScratchReg, ScratchRegister); masm.ma_b(ScratchRegister, ScratchRegister, &again, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); masm.bind(&end); return; } masm.andi(offsetTemp, SecondScratchReg, 3); masm.subPtr(offsetTemp, SecondScratchReg); #if !MOZ_LITTLE_ENDIAN() masm.as_xori(offsetTemp, offsetTemp, 3); #endif masm.slli(offsetTemp, offsetTemp, 3); masm.ma_li(maskTemp, Imm32(UINT32_MAX >> ((4 - nbytes) * 8))); masm.sll(maskTemp, maskTemp, offsetTemp); masm.nor(maskTemp, zero, maskTemp); masm.memoryBarrierBefore(sync); masm.bind(&again); if (access) { masm.append(*access, wasm::TrapMachineInsn::Atomic, FaultingCodeOffset(masm.currentOffset())); } masm.lr_w(true, true, ScratchRegister, SecondScratchReg); masm.srl(output, ScratchRegister, offsetTemp); switch (nbytes) { case 1: if (signExtend) { masm.SignExtendByte(valueTemp, oldval); masm.SignExtendByte(output, output); } else { masm.andi(valueTemp, oldval, 0xff); masm.andi(output, output, 0xff); } break; case 2: if (signExtend) { masm.SignExtendShort(valueTemp, oldval); masm.SignExtendShort(output, output); } else { masm.andi(valueTemp, oldval, 0xffff); masm.andi(output, output, 0xffff); } break; } masm.ma_b(output, valueTemp, &end, Assembler::NotEqual, ShortJump); masm.sll(valueTemp, newval, offsetTemp); masm.and_(ScratchRegister, ScratchRegister, maskTemp); masm.or_(ScratchRegister, ScratchRegister, valueTemp); masm.sc_w(true, true, ScratchRegister, SecondScratchReg, ScratchRegister); masm.ma_b(ScratchRegister, ScratchRegister, &again, Assembler::NonZero, ShortJump); masm.memoryBarrierAfter(sync); masm.bind(&end); } void MacroAssembler::compareExchange(Scalar::Type type, Synchronization sync, const Address& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { CompareExchange(*this, nullptr, type, sync, mem, oldval, newval, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::compareExchange(Scalar::Type type, Synchronization sync, const BaseIndex& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { CompareExchange(*this, nullptr, type, sync, mem, oldval, newval, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::wasmCompareExchange(const wasm::MemoryAccessDesc& access, const Address& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { CompareExchange(*this, &access, access.type(), access.sync(), mem, oldval, newval, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::wasmCompareExchange(const wasm::MemoryAccessDesc& access, const BaseIndex& mem, Register oldval, Register newval, Register valueTemp, Register offsetTemp, Register maskTemp, Register output) { CompareExchange(*this, &access, access.type(), access.sync(), mem, oldval, newval, valueTemp, offsetTemp, maskTemp, output); } void MacroAssembler::wasmLoad(const wasm::MemoryAccessDesc& access, Register memoryBase, Register ptr, Register ptrScratch, AnyRegister output) { wasmLoadImpl(access, memoryBase, ptr, ptrScratch, output, InvalidReg); } void MacroAssembler::wasmLoadI64(const wasm::MemoryAccessDesc& access, Register memoryBase, Register ptr, Register ptrScratch, Register64 output) { wasmLoadI64Impl(access, memoryBase, ptr, ptrScratch, output, InvalidReg); } void MacroAssembler::wasmStore(const wasm::MemoryAccessDesc& access, AnyRegister value, Register memoryBase, Register ptr, Register ptrScratch) { wasmStoreImpl(access, value, memoryBase, ptr, ptrScratch, InvalidReg); } void MacroAssembler::wasmStoreI64(const wasm::MemoryAccessDesc& access, Register64 value, Register memoryBase, Register ptr, Register ptrScratch) { wasmStoreI64Impl(access, value, memoryBase, ptr, ptrScratch, InvalidReg); } void MacroAssemblerRiscv64::Clear_if_nan_d(Register rd, FPURegister fs) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Label no_nan; feq_d(ScratchRegister, fs, fs); bnez(ScratchRegister, &no_nan); mv(rd, zero_reg); bind(&no_nan); } void MacroAssemblerRiscv64::Clear_if_nan_s(Register rd, FPURegister fs) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Label no_nan; feq_s(ScratchRegister, fs, fs); bnez(ScratchRegister, &no_nan); mv(rd, zero_reg); bind(&no_nan); } void MacroAssembler::wasmTruncateDoubleToInt32(FloatRegister input, Register output, bool isSaturating, Label* oolEntry) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_w_d(output, input, ScratchRegister); if (isSaturating) { Clear_if_nan_d(output, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } void MacroAssembler::wasmTruncateDoubleToInt64( FloatRegister input, Register64 output, bool isSaturating, Label* oolEntry, Label* oolRejoin, FloatRegister tempDouble) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_l_d(output.reg, input, ScratchRegister); if (isSaturating) { bind(oolRejoin); Clear_if_nan_d(output.reg, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } void MacroAssembler::wasmTruncateDoubleToUInt32(FloatRegister input, Register output, bool isSaturating, Label* oolEntry) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_uw_d(output, input, ScratchRegister); if (isSaturating) { Clear_if_nan_d(output, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } void MacroAssembler::wasmTruncateDoubleToUInt64( FloatRegister input, Register64 output, bool isSaturating, Label* oolEntry, Label* oolRejoin, FloatRegister tempDouble) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_ul_d(output.reg, input, ScratchRegister); if (isSaturating) { bind(oolRejoin); Clear_if_nan_d(output.reg, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } void MacroAssembler::wasmTruncateFloat32ToInt32(FloatRegister input, Register output, bool isSaturating, Label* oolEntry) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_w_s(output, input, ScratchRegister); if (isSaturating) { Clear_if_nan_s(output, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } void MacroAssembler::wasmTruncateFloat32ToInt64( FloatRegister input, Register64 output, bool isSaturating, Label* oolEntry, Label* oolRejoin, FloatRegister tempFloat) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_l_s(output.reg, input, ScratchRegister); if (isSaturating) { bind(oolRejoin); Clear_if_nan_s(output.reg, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } void MacroAssembler::wasmTruncateFloat32ToUInt32(FloatRegister input, Register output, bool isSaturating, Label* oolEntry) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_uw_s(output, input, ScratchRegister); if (isSaturating) { Clear_if_nan_s(output, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } void MacroAssembler::wasmTruncateFloat32ToUInt64( FloatRegister input, Register64 output, bool isSaturating, Label* oolEntry, Label* oolRejoin, FloatRegister tempFloat) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); Trunc_ul_s(output.reg, input, ScratchRegister); if (isSaturating) { bind(oolRejoin); Clear_if_nan_s(output.reg, input); } else { ma_b(ScratchRegister, Imm32(1), oolEntry, Assembler::NotEqual); } } // TODO(riscv64): widenInt32 should be nop? void MacroAssembler::widenInt32(Register r) { move32To64SignExtend(r, Register64(r)); } #ifdef ENABLE_WASM_TAIL_CALLS void MacroAssembler::wasmMarkSlowCall() { mv(ra, ra); } const int32_t SlowCallMarker = 0x8093; // addi ra, ra, 0 void MacroAssembler::wasmCheckSlowCallsite(Register ra_, Label* notSlow, Register temp1, Register temp2) { MOZ_ASSERT(ra_ != temp2); load32(Address(ra_, 0), temp2); branch32(Assembler::NotEqual, temp2, Imm32(SlowCallMarker), notSlow); } #endif // ENABLE_WASM_TAIL_CALLS //}}} check_macroassembler_style // This method generates lui, dsll and ori instruction block that can be // modified by UpdateLoad64Value, either during compilation (eg. // Assembler::bind), or during execution (eg. jit::PatchJump). void MacroAssemblerRiscv64::ma_liPatchable(Register dest, Imm32 imm) { m_buffer.ensureSpace(2 * sizeof(uint32_t)); int64_t value = imm.value; int64_t high_20 = ((value + 0x800) >> 12); int64_t low_12 = value << 52 >> 52; lui(dest, high_20); addi(dest, dest, low_12); } void MacroAssemblerRiscv64::ma_liPatchable(Register dest, ImmPtr imm) { return ma_liPatchable(dest, ImmWord(uintptr_t(imm.value))); } void MacroAssemblerRiscv64::ma_liPatchable(Register dest, ImmWord imm, LiFlags flags) { DEBUG_PRINTF("\tma_liPatchable\n"); if (Li64 == flags) { li_constant(dest, imm.value); } else { li_ptr(dest, imm.value); } } void MacroAssemblerRiscv64::ma_li(Register dest, ImmGCPtr ptr) { BlockTrampolinePoolScope block_trampoline_pool(this, 6); writeDataRelocation(ptr); ma_liPatchable(dest, ImmPtr(ptr.value)); } void MacroAssemblerRiscv64::ma_li(Register dest, Imm32 imm) { RV_li(dest, imm.value); } void MacroAssemblerRiscv64::ma_li(Register dest, Imm64 imm) { RV_li(dest, imm.value); } void MacroAssemblerRiscv64::ma_li(Register dest, CodeLabel* label) { DEBUG_PRINTF("[ %s\n", __FUNCTION__); BlockTrampolinePoolScope block_trampoline_pool(this, 7); BufferOffset bo = m_buffer.nextOffset(); JitSpew(JitSpew_Codegen, ".load CodeLabel %p", label); ma_liPatchable(dest, ImmWord(/* placeholder */ 0)); label->patchAt()->bind(bo.getOffset()); label->setLinkMode(CodeLabel::MoveImmediate); DEBUG_PRINTF("]\n"); } void MacroAssemblerRiscv64::ma_li(Register dest, ImmWord imm) { RV_li(dest, imm.value); } // Shortcut for when we know we're transferring 32 bits of data. void MacroAssemblerRiscv64::ma_pop(Register r) { ld(r, StackPointer, 0); addi(StackPointer, StackPointer, sizeof(intptr_t)); } void MacroAssemblerRiscv64::ma_push(Register r) { if (r == sp) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); // Pushing sp requires one more instruction. mv(ScratchRegister, sp); r = ScratchRegister; } addi(StackPointer, StackPointer, (int32_t) - sizeof(intptr_t)); sd(r, StackPointer, 0); } // multiplies. For now, there are only few that we care about. void MacroAssemblerRiscv64::ma_mul32TestOverflow(Register rd, Register rj, Register rk, Label* overflow) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); MulOverflow32(rd, rj, rk, ScratchRegister); ma_b(ScratchRegister, Register(zero), overflow, Assembler::NotEqual); } void MacroAssemblerRiscv64::ma_mul32TestOverflow(Register rd, Register rj, Imm32 imm, Label* overflow) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); MulOverflow32(rd, rj, Operand(imm.value), ScratchRegister); ma_b(ScratchRegister, Register(zero), overflow, Assembler::NotEqual); } void MacroAssemblerRiscv64::ma_mulPtrTestOverflow(Register rd, Register rj, Register rk, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); Register scratch2 = temps.Acquire(); MOZ_ASSERT(rd != scratch); if (rd == rj) { or_(scratch, rj, zero); rj = scratch; rk = (rd == rk) ? rj : rk; } else if (rd == rk) { or_(scratch, rk, zero); rk = scratch; } mul(rd, rj, rk); mulh(scratch, rj, rk); srai(scratch2, rd, 63); ma_b(scratch, Register(scratch2), overflow, Assembler::NotEqual); } // MulOverflow32 sets overflow register to zero if no overflow occured void MacroAssemblerRiscv64::MulOverflow32(Register dst, Register left, const Operand& right, Register overflow) { UseScratchRegisterScope temps(this); BlockTrampolinePoolScope block_trampoline_pool(this, 11); Register right_reg; Register scratch = temps.Acquire(); Register scratch2 = temps.Acquire(); if (right.is_imm()) { ma_li(scratch, right.immediate()); right_reg = scratch; } else { MOZ_ASSERT(right.is_reg()); right_reg = right.rm(); } MOZ_ASSERT(left != scratch2 && right_reg != scratch2 && dst != scratch2 && overflow != scratch2); MOZ_ASSERT(overflow != left && overflow != right_reg); sext_w(overflow, left); sext_w(scratch2, right_reg); mul(overflow, overflow, scratch2); sext_w(dst, overflow); xor_(overflow, overflow, dst); } int32_t MacroAssemblerRiscv64::GetOffset(int32_t offset, Label* L, OffsetSize bits) { if (L) { offset = branch_offset_helper(L, bits); } else { MOZ_ASSERT(is_intn(offset, bits)); } return offset; } bool MacroAssemblerRiscv64::CalculateOffset(Label* L, int32_t* offset, OffsetSize bits) { if (!is_near(L, bits)) return false; *offset = GetOffset(*offset, L, bits); return true; } void MacroAssemblerRiscv64::BranchShortHelper(int32_t offset, Label* L) { MOZ_ASSERT(L == nullptr || offset == 0); offset = GetOffset(offset, L, OffsetSize::kOffset21); Assembler::j(offset); } bool MacroAssemblerRiscv64::BranchShortHelper(int32_t offset, Label* L, Condition cond, Register rs, const Operand& rt) { MOZ_ASSERT(L == nullptr || offset == 0); MOZ_ASSERT(rt.is_reg() || rt.is_imm()); UseScratchRegisterScope temps(this); Register scratch = Register(); if (rt.is_imm()) { scratch = temps.Acquire(); ma_li(scratch, Imm64(rt.immediate())); } else { MOZ_ASSERT(rt.is_reg()); scratch = rt.rm(); } BlockTrampolinePoolScope block_trampoline_pool(this, 2); { switch (cond) { case Always: if (!CalculateOffset(L, &offset, OffsetSize::kOffset21)) return false; Assembler::j(offset); EmitConstPoolWithJumpIfNeeded(); break; case Equal: // rs == rt if (rt.is_reg() && rs == rt.rm()) { if (!CalculateOffset(L, &offset, OffsetSize::kOffset21)) return false; Assembler::j(offset); } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::beq(rs, scratch, offset); } break; case NotEqual: // rs != rt if (rt.is_reg() && rs == rt.rm()) { break; // No code needs to be emitted } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::bne(rs, scratch, offset); } break; // Signed comparison. case GreaterThan: // rs > rt if (rt.is_reg() && rs == rt.rm()) { break; // No code needs to be emitted. } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::bgt(rs, scratch, offset); } break; case GreaterThanOrEqual: // rs >= rt if (rt.is_reg() && rs == rt.rm()) { if (!CalculateOffset(L, &offset, OffsetSize::kOffset21)) return false; Assembler::j(offset); } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::bge(rs, scratch, offset); } break; case LessThan: // rs < rt if (rt.is_reg() && rs == rt.rm()) { break; // No code needs to be emitted. } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::blt(rs, scratch, offset); } break; case LessThanOrEqual: // rs <= rt if (rt.is_reg() && rs == rt.rm()) { if (!CalculateOffset(L, &offset, OffsetSize::kOffset21)) return false; Assembler::j(offset); } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::ble(rs, scratch, offset); } break; // Unsigned comparison. case Above: // rs > rt if (rt.is_reg() && rs == rt.rm()) { break; // No code needs to be emitted. } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::bgtu(rs, scratch, offset); } break; case AboveOrEqual: // rs >= rt if (rt.is_reg() && rs == rt.rm()) { if (!CalculateOffset(L, &offset, OffsetSize::kOffset21)) return false; Assembler::j(offset); } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::bgeu(rs, scratch, offset); } break; case Below: // rs < rt if (rt.is_reg() && rs == rt.rm()) { break; // No code needs to be emitted. } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; bltu(rs, scratch, offset); } break; case BelowOrEqual: // rs <= rt if (rt.is_reg() && rs == rt.rm()) { if (!CalculateOffset(L, &offset, OffsetSize::kOffset21)) return false; Assembler::j(offset); } else { if (!CalculateOffset(L, &offset, OffsetSize::kOffset13)) return false; Assembler::bleu(rs, scratch, offset); } break; default: MOZ_CRASH("UNREACHABLE"); } } return true; } // BRANCH_ARGS_CHECK checks that conditional jump arguments are correct. #define BRANCH_ARGS_CHECK(cond, rs, rt) \ MOZ_ASSERT((cond == Always && rs == zero && rt.rm() == zero) || \ (cond != Always && (rs != zero || rt.rm() != zero))) bool MacroAssemblerRiscv64::BranchShortCheck(int32_t offset, Label* L, Condition cond, Register rs, const Operand& rt) { BRANCH_ARGS_CHECK(cond, rs, rt); if (!L) { MOZ_ASSERT(is_int13(offset)); return BranchShortHelper(offset, nullptr, cond, rs, rt); } else { MOZ_ASSERT(offset == 0); return BranchShortHelper(0, L, cond, rs, rt); } } void MacroAssemblerRiscv64::BranchShort(Label* L) { BranchShortHelper(0, L); } void MacroAssemblerRiscv64::BranchShort(int32_t offset, Condition cond, Register rs, const Operand& rt) { BranchShortCheck(offset, nullptr, cond, rs, rt); } void MacroAssemblerRiscv64::BranchShort(Label* L, Condition cond, Register rs, const Operand& rt) { BranchShortCheck(0, L, cond, rs, rt); } void MacroAssemblerRiscv64::BranchLong(Label* L) { // Generate position independent long branch. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); int32_t imm; imm = branch_long_offset(L); GenPCRelativeJump(scratch, imm); } void MacroAssemblerRiscv64::BranchAndLinkLong(Label* L) { // Generate position independent long branch and link. int32_t imm; imm = branch_long_offset(L); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); GenPCRelativeJumpAndLink(scratch, imm); } void MacroAssemblerRiscv64::ma_branch(Label* L, Condition cond, Register rs, const Operand& rt, JumpKind jumpKind) { if (L->used()) { if (jumpKind == ShortJump && BranchShortCheck(0, L, cond, rs, rt)) { return; } if (cond != Always) { Label skip; Condition neg_cond = InvertCondition(cond); BranchShort(&skip, neg_cond, rs, rt); BranchLong(L); bind(&skip); } else { BranchLong(L); EmitConstPoolWithJumpIfNeeded(); } } else { if (jumpKind == LongJump) { if (cond != Always) { Label skip; Condition neg_cond = InvertCondition(cond); BranchShort(&skip, neg_cond, rs, rt); BranchLong(L); bind(&skip); } else { BranchLong(L); EmitConstPoolWithJumpIfNeeded(); } } else { BranchShort(L, cond, rs, rt); } } } // Branches when done from within riscv code. void MacroAssemblerRiscv64::ma_b(Register lhs, Address addr, Label* label, Condition c, JumpKind jumpKind) { ScratchRegisterScope scratch(asMasm()); MOZ_ASSERT(lhs != scratch); ma_load(scratch, addr, SizeDouble); ma_b(lhs, Register(scratch), label, c, jumpKind); } void MacroAssemblerRiscv64::ma_b(Register lhs, ImmPtr imm, Label* l, Condition c, JumpKind jumpKind) { asMasm().ma_b(lhs, ImmWord(uintptr_t(imm.value)), l, c, jumpKind); } // Branches when done from within loongarch-specific code. void MacroAssemblerRiscv64::ma_b(Register lhs, ImmWord imm, Label* label, Condition c, JumpKind jumpKind) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(lhs != scratch); ma_li(scratch, imm); ma_b(lhs, Register(scratch), label, c, jumpKind); } void MacroAssemblerRiscv64::ma_b(Register lhs, Imm32 imm, Label* label, Condition c, JumpKind jumpKind) { if ((c == NonZero || c == Zero || c == Signed || c == NotSigned) && imm.value == 0) { ma_b(lhs, lhs, label, c, jumpKind); } else { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(lhs != scratch); ma_li(scratch, imm); ma_b(lhs, Register(scratch), label, c, jumpKind); } } void MacroAssemblerRiscv64::ma_b(Address addr, Imm32 imm, Label* label, Condition c, JumpKind jumpKind) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); ma_load(scratch2, addr); ma_b(Register(scratch2), imm, label, c, jumpKind); } void MacroAssemblerRiscv64::ma_b(Register lhs, Register rhs, Label* label, Condition c, JumpKind jumpKind) { switch (c) { case Equal: case NotEqual: ma_branch(label, c, lhs, rhs, jumpKind); break; case Always: ma_branch(label, c, zero, Operand(zero), jumpKind); break; case Zero: MOZ_ASSERT(lhs == rhs); ma_branch(label, Equal, lhs, Operand(zero), jumpKind); break; case NonZero: MOZ_ASSERT(lhs == rhs); ma_branch(label, NotEqual, lhs, Operand(zero), jumpKind); break; case Signed: MOZ_ASSERT(lhs == rhs); ma_branch(label, LessThan, lhs, Operand(zero), jumpKind); break; case NotSigned: MOZ_ASSERT(lhs == rhs); ma_branch(label, GreaterThanOrEqual, lhs, Operand(zero), jumpKind); break; default: { ma_branch(label, c, lhs, rhs, jumpKind); break; } } } void MacroAssemblerRiscv64::ExtractBits(Register rt, Register rs, uint16_t pos, uint16_t size, bool sign_extend) { #if JS_CODEGEN_RISCV64 MOZ_ASSERT(pos < 64 && 0 < size && size <= 64 && 0 < pos + size && pos + size <= 64); slli(rt, rs, 64 - (pos + size)); if (sign_extend) { srai(rt, rt, 64 - size); } else { srli(rt, rt, 64 - size); } #elif JS_CODEGEN_RISCV32 MOZ_ASSERT(pos < 32); MOZ_ASSERT(size > 0); MOZ_ASSERT(size <= 32); MOZ_ASSERT((pos + size) > 0); MOZ_ASSERT((pos + size) <= 32); slli(rt, rs, 32 - (pos + size)); if (sign_extend) { srai(rt, rt, 32 - size); } else { srli(rt, rt, 32 - size); } #endif } void MacroAssemblerRiscv64::InsertBits(Register dest, Register source, int pos, int size) { #if JS_CODEGEN_RISCV64 MOZ_ASSERT(size < 64); #elif JS_CODEGEN_RISCV32 MOZ_ASSERT(size < 32); #endif UseScratchRegisterScope temps(this); Register mask = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); Register source_ = temps.Acquire(); // Create a mask of the length=size. ma_li(mask, Imm32(1)); slli(mask, mask, size); addi(mask, mask, -1); and_(source_, mask, source); slli(source_, source_, pos); // Make a mask containing 0's. 0's start at "pos" with length=size. slli(mask, mask, pos); not_(mask, mask); // cut area for insertion of source. and_(dest, mask, dest); // insert source or_(dest, dest, source_); } void MacroAssemblerRiscv64::InsertBits(Register dest, Register source, Register pos, int size) { #if JS_CODEGEN_RISCV64 MOZ_ASSERT(size < 64); #elif JS_CODEGEN_RISCV32 MOZ_ASSERT(size < 32); #endif UseScratchRegisterScope temps(this); Register mask = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); Register source_ = temps.Acquire(); // Create a mask of the length=size. ma_li(mask, Imm32(1)); slli(mask, mask, size); addi(mask, mask, -1); and_(source_, mask, source); sll(source_, source_, pos); // Make a mask containing 0's. 0's start at "pos" with length=size. sll(mask, mask, pos); not_(mask, mask); // cut area for insertion of source. and_(dest, mask, dest); // insert source or_(dest, dest, source_); } void MacroAssemblerRiscv64::ma_add32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { if (is_int12(rt.immediate())) { addiw(rd, rs, static_cast(rt.immediate())); } else if ((-4096 <= rt.immediate() && rt.immediate() <= -2049) || (2048 <= rt.immediate() && rt.immediate() <= 4094)) { addiw(rd, rs, rt.immediate() / 2); addiw(rd, rd, rt.immediate() - (rt.immediate() / 2)); } else { // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); addw(rd, rs, scratch); } } else { MOZ_ASSERT(rt.is_reg()); addw(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_add64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { if (is_int12(rt.immediate())) { addi(rd, rs, static_cast(rt.immediate())); } else if ((-4096 <= rt.immediate() && rt.immediate() <= -2049) || (2048 <= rt.immediate() && rt.immediate() <= 4094)) { addi(rd, rs, rt.immediate() / 2); addi(rd, rd, rt.immediate() - (rt.immediate() / 2)); } else { // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); add(rd, rs, scratch); } } else { MOZ_ASSERT(rt.is_reg()); add(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_sub32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { if (is_int12(-rt.immediate())) { addiw(rd, rs, static_cast( -rt.immediate())); // No subi instr, use addi(x, y, -imm). } else if ((-4096 <= -rt.immediate() && -rt.immediate() <= -2049) || (2048 <= -rt.immediate() && -rt.immediate() <= 4094)) { addiw(rd, rs, -rt.immediate() / 2); addiw(rd, rd, -rt.immediate() - (-rt.immediate() / 2)); } else { // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, rt.immediate()); subw(rd, rs, scratch); } } else { MOZ_ASSERT(rt.is_reg()); subw(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_sub64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { if (is_int12(-rt.immediate())) { addi(rd, rs, static_cast( -rt.immediate())); // No subi instr, use addi(x, y, -imm). } else if ((-4096 <= -rt.immediate() && -rt.immediate() <= -2049) || (2048 <= -rt.immediate() && -rt.immediate() <= 4094)) { addi(rd, rs, -rt.immediate() / 2); addi(rd, rd, -rt.immediate() - (-rt.immediate() / 2)); } else { // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, rt.immediate()); sub(rd, rs, scratch); } } else { MOZ_ASSERT(rt.is_reg()); sub(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_and(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { if (is_int12(rt.immediate())) { andi(rd, rs, rt.immediate()); } else { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); and_(rd, rs, ScratchRegister); } } else { MOZ_ASSERT(rt.is_reg()); and_(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_or(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { if (is_int12(rt.immediate())) { ori(rd, rs, rt.immediate()); } else { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); or_(rd, rs, ScratchRegister); } } else { MOZ_ASSERT(rt.is_reg()); or_(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_xor(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { if (is_int12(rt.immediate())) { xori(rd, rs, rt.immediate()); } else { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); xor_(rd, rs, ScratchRegister); } } else { MOZ_ASSERT(rt.is_reg()); xor_(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_nor(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); nor(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); nor(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_div32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); divw(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); divw(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_divu32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); divuw(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); divuw(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_div64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); div(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); div(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_divu64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); divu(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); divu(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_mod32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); remw(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); remw(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_modu32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); remuw(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); remuw(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_mod64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); rem(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); rem(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_modu64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); remu(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); remu(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_mul32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); mulw(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); mulw(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_mulh32(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); mul(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); mul(rd, rs, rt.rm()); } srai(rd, rd, 32); } void MacroAssemblerRiscv64::ma_mul64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); mul(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); mul(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_mulh64(Register rd, Register rs, Operand rt) { if (rt.is_imm()) { UseScratchRegisterScope temps(this); Register ScratchRegister = temps.Acquire(); ma_li(ScratchRegister, rt.immediate()); mulh(rd, rs, ScratchRegister); } else { MOZ_ASSERT(rt.is_reg()); mulh(rd, rs, rt.rm()); } } void MacroAssemblerRiscv64::ma_sll64(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { sll(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); uint8_t shamt = static_cast(rt.immediate()); slli(rd, rs, shamt); } } void MacroAssemblerRiscv64::ma_sll32(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { sllw(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); uint8_t shamt = static_cast(rt.immediate()); slliw(rd, rs, shamt); } } void MacroAssemblerRiscv64::ma_sra64(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { sra(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); uint8_t shamt = static_cast(rt.immediate()); srai(rd, rs, shamt); } } void MacroAssemblerRiscv64::ma_sra32(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { sraw(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); uint8_t shamt = static_cast(rt.immediate()); sraiw(rd, rs, shamt); } } void MacroAssemblerRiscv64::ma_srl64(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { srl(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); uint8_t shamt = static_cast(rt.immediate()); srli(rd, rs, shamt); } } void MacroAssemblerRiscv64::ma_srl32(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { srlw(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); uint8_t shamt = static_cast(rt.immediate()); srliw(rd, rs, shamt); } } void MacroAssemblerRiscv64::ma_slt(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { slt(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); if (is_int12(rt.immediate())) { slti(rd, rs, static_cast(rt.immediate())); } else { // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); slt(rd, rs, scratch); } } } void MacroAssemblerRiscv64::ma_sltu(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { sltu(rd, rs, rt.rm()); } else { MOZ_ASSERT(rt.is_imm()); if (is_int12(rt.immediate())) { sltiu(rd, rs, static_cast(rt.immediate())); } else { // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); sltu(rd, rs, scratch); } } } void MacroAssemblerRiscv64::ma_sle(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { slt(rd, rt.rm(), rs); } else { MOZ_ASSERT(rt.is_imm()); // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); slt(rd, scratch, rs); } xori(rd, rd, 1); } void MacroAssemblerRiscv64::ma_sleu(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { sltu(rd, rt.rm(), rs); } else { MOZ_ASSERT(rt.is_imm()); // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); sltu(rd, scratch, rs); } xori(rd, rd, 1); } void MacroAssemblerRiscv64::ma_sgt(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { slt(rd, rt.rm(), rs); } else { MOZ_ASSERT(rt.is_imm()); // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); slt(rd, scratch, rs); } } void MacroAssemblerRiscv64::ma_sgtu(Register rd, Register rs, Operand rt) { if (rt.is_reg()) { sltu(rd, rt.rm(), rs); } else { MOZ_ASSERT(rt.is_imm()); // li handles the relocation. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 9); ma_li(scratch, rt.immediate()); sltu(rd, scratch, rs); } } void MacroAssemblerRiscv64::ma_sge(Register rd, Register rs, Operand rt) { ma_slt(rd, rs, rt); xori(rd, rd, 1); } void MacroAssemblerRiscv64::ma_sgeu(Register rd, Register rs, Operand rt) { ma_sltu(rd, rs, rt); xori(rd, rd, 1); } static inline bool IsZero(const Operand& rt) { if (rt.is_reg()) { return rt.rm() == zero_reg; } else { MOZ_ASSERT(rt.is_imm()); return rt.immediate() == 0; } } void MacroAssemblerRiscv64::ma_seq(Register rd, Register rs, Operand rt) { if (rs == zero_reg) { ma_seqz(rd, rt); } else if (IsZero(rt)) { seqz(rd, rs); } else { ma_sub64(rd, rs, rt); seqz(rd, rd); } } void MacroAssemblerRiscv64::ma_sne(Register rd, Register rs, Operand rt) { if (rs == zero_reg) { ma_snez(rd, rt); } else if (IsZero(rt)) { snez(rd, rs); } else { ma_sub64(rd, rs, rt); snez(rd, rd); } } void MacroAssemblerRiscv64::ma_seqz(Register rd, const Operand& rt) { if (rt.is_reg()) { seqz(rd, rt.rm()); } else { ma_li(rd, rt.immediate() == 0); } } void MacroAssemblerRiscv64::ma_snez(Register rd, const Operand& rt) { if (rt.is_reg()) { snez(rd, rt.rm()); } else { ma_li(rd, rt.immediate() != 0); } } void MacroAssemblerRiscv64::ma_neg(Register rd, const Operand& rt) { MOZ_ASSERT(rt.is_reg()); neg(rd, rt.rm()); } void MacroAssemblerRiscv64::ma_jump(ImmPtr dest) { DEBUG_PRINTF("[ %s\n", __FUNCTION__); BlockTrampolinePoolScope block_trampoline_pool(this, 8); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); asMasm().ma_liPatchable(scratch, dest); jr(scratch, 0); DEBUG_PRINTF("]\n"); } // fp instructions void MacroAssemblerRiscv64::ma_lid(FloatRegister dest, double value) { ImmWord imm(mozilla::BitwiseCast(value)); if (imm.value != 0) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, imm); fmv_d_x(dest, scratch); } else { fmv_d_x(dest, zero); } } // fp instructions void MacroAssemblerRiscv64::ma_lis(FloatRegister dest, float value) { Imm32 imm(mozilla::BitwiseCast(value)); if (imm.value != 0) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, imm); fmv_w_x(dest, scratch); } else { fmv_w_x(dest, zero); } } void MacroAssemblerRiscv64::ma_sub32TestOverflow(Register rd, Register rj, Register rk, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); sub(scratch, rj, rk); subw(rd, rj, rk); ma_b(rd, Register(scratch), overflow, Assembler::NotEqual); } void MacroAssemblerRiscv64::ma_sub32TestOverflow(Register rd, Register rj, Imm32 imm, Label* overflow) { if (imm.value != INT32_MIN) { asMasm().ma_add32TestOverflow(rd, rj, Imm32(-imm.value), overflow); } else { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(rj != scratch); ma_li(scratch, Imm32(imm.value)); asMasm().ma_sub32TestOverflow(rd, rj, scratch, overflow); } } void MacroAssemblerRiscv64::ma_add32TestOverflow(Register rd, Register rj, Register rk, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); add(scratch, rj, rk); addw(rd, rj, rk); ma_b(rd, Register(scratch), overflow, Assembler::NotEqual); } void MacroAssemblerRiscv64::ma_add32TestOverflow(Register rd, Register rj, Imm32 imm, Label* overflow) { // Check for signed range because of addi if (is_intn(imm.value, 12)) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); addi(scratch, rj, imm.value); addiw(rd, rj, imm.value); ma_b(rd, scratch, overflow, Assembler::NotEqual); } else { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); ma_li(scratch2, imm); ma_add32TestOverflow(rd, rj, scratch2, overflow); } } void MacroAssemblerRiscv64::ma_subPtrTestOverflow(Register rd, Register rj, Register rk, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); MOZ_ASSERT_IF(rj == rd, rj != rk); MOZ_ASSERT(rj != scratch2); MOZ_ASSERT(rk != scratch2); MOZ_ASSERT(rd != scratch2); Register rj_copy = rj; if (rj == rd) { ma_or(scratch2, rj, zero); rj_copy = scratch2; } { Register scratch = temps.Acquire(); MOZ_ASSERT(rd != scratch); sub(rd, rj, rk); // If the sign of rj and rk are the same, no overflow ma_xor(scratch, rj_copy, rk); // Check if the sign of rd and rj are the same ma_xor(scratch2, rd, rj_copy); ma_and(scratch2, scratch2, scratch); } ma_b(scratch2, zero, overflow, Assembler::LessThan); } void MacroAssemblerRiscv64::ma_addPtrTestOverflow(Register rd, Register rj, Register rk, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(rd != scratch); if (rj == rk) { if (rj == rd) { ma_or(scratch, rj, zero); rj = scratch; } add(rd, rj, rj); ma_xor(scratch, rj, rd); ma_b(scratch, zero, overflow, Assembler::LessThan); } else { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); MOZ_ASSERT(rj != scratch); MOZ_ASSERT(rd != scratch2); if (rj == rd) { ma_or(scratch2, rj, zero); rj = scratch2; } add(rd, rj, rk); slti(scratch, rj, 0); slt(scratch2, rd, rj); ma_b(scratch, Register(scratch2), overflow, Assembler::NotEqual); } } void MacroAssemblerRiscv64::ma_addPtrTestOverflow(Register rd, Register rj, Imm32 imm, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); if (imm.value == 0) { ori(rd, rj, 0); return; } if (rj == rd) { ori(scratch2, rj, 0); rj = scratch2; } ma_add64(rd, rj, imm); if (imm.value > 0) { ma_b(rd, rj, overflow, Assembler::LessThan); } else { MOZ_ASSERT(imm.value < 0); ma_b(rd, rj, overflow, Assembler::GreaterThan); } } void MacroAssemblerRiscv64::ma_addPtrTestOverflow(Register rd, Register rj, ImmWord imm, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); if (imm.value == 0) { ori(rd, rj, 0); return; } if (rj == rd) { MOZ_ASSERT(rj != scratch2); ori(scratch2, rj, 0); rj = scratch2; } ma_li(rd, imm); add(rd, rj, rd); if (imm.value > 0) { ma_b(rd, rj, overflow, Assembler::LessThan); } else { MOZ_ASSERT(imm.value < 0); ma_b(rd, rj, overflow, Assembler::GreaterThan); } } void MacroAssemblerRiscv64::ma_add32TestCarry(Condition cond, Register rd, Register rj, Register rk, Label* overflow) { MOZ_ASSERT(cond == Assembler::CarrySet || cond == Assembler::CarryClear); MOZ_ASSERT_IF(rd == rj, rk != rd); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); addw(rd, rj, rk); sltu(scratch, rd, rd == rj ? rk : rj); ma_b(Register(scratch), Register(scratch), overflow, cond == Assembler::CarrySet ? Assembler::NonZero : Assembler::Zero); } void MacroAssemblerRiscv64::ma_add32TestCarry(Condition cond, Register rd, Register rj, Imm32 imm, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); MOZ_ASSERT(rj != scratch2); ma_li(scratch2, imm); ma_add32TestCarry(cond, rd, rj, scratch2, overflow); } void MacroAssemblerRiscv64::ma_subPtrTestOverflow(Register rd, Register rj, Imm32 imm, Label* overflow) { // TODO(riscv): Check subPtrTestOverflow MOZ_ASSERT(imm.value != INT32_MIN); ma_addPtrTestOverflow(rd, rj, Imm32(-imm.value), overflow); } void MacroAssemblerRiscv64::ma_addPtrTestCarry(Condition cond, Register rd, Register rj, Register rk, Label* label) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(rd != rk); MOZ_ASSERT(rd != scratch); add(rd, rj, rk); sltu(scratch, rd, rk); ma_b(scratch, Register(scratch), label, cond == Assembler::CarrySet ? Assembler::NonZero : Assembler::Zero); } void MacroAssemblerRiscv64::ma_addPtrTestCarry(Condition cond, Register rd, Register rj, Imm32 imm, Label* label) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); // Check for signed range because of addi if (is_intn(imm.value, 12)) { addi(rd, rj, imm.value); sltiu(scratch2, rd, imm.value); ma_b(scratch2, scratch2, label, cond == Assembler::CarrySet ? Assembler::NonZero : Assembler::Zero); } else { ma_li(scratch2, imm); ma_addPtrTestCarry(cond, rd, rj, scratch2, label); } } void MacroAssemblerRiscv64::ma_addPtrTestCarry(Condition cond, Register rd, Register rj, ImmWord imm, Label* label) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); // Check for signed range because of addi_d if (is_intn(imm.value, 12)) { uint32_t value = imm.value; addi(rd, rj, value); ma_sltu(scratch2, rd, Operand(value)); ma_b(scratch2, scratch2, label, cond == Assembler::CarrySet ? Assembler::NonZero : Assembler::Zero); } else { ma_li(scratch2, imm); ma_addPtrTestCarry(cond, rd, rj, scratch2, label); } } FaultingCodeOffset MacroAssemblerRiscv64::ma_load( Register dest, const BaseIndex& src, LoadStoreSize size, LoadStoreExtension extension) { UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); asMasm().computeScaledAddress(src, scratch2); return asMasm().ma_load(dest, Address(scratch2, src.offset), size, extension); } void MacroAssemblerRiscv64::ma_pop(FloatRegister f) { fld(f, StackPointer, 0); addi(StackPointer, StackPointer, sizeof(double)); } void MacroAssemblerRiscv64::ma_push(FloatRegister f) { addi(StackPointer, StackPointer, (int32_t) - sizeof(double)); fsd(f, StackPointer, 0); } FaultingCodeOffset MacroAssemblerRiscv64::ma_fld_s(FloatRegister ft, Address address) { int32_t offset = address.offset; Register base = address.base; FaultingCodeOffset fco; if (is_intn(offset, 12)) { fco = FaultingCodeOffset(currentOffset()); flw(ft, base, offset); } else { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(base != scratch); ma_li(scratch, Imm32(offset)); ma_add64(scratch, base, scratch); fco = FaultingCodeOffset(currentOffset()); flw(ft, scratch, 0); } return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_fld_d(FloatRegister ft, Address address) { int32_t offset = address.offset; Register base = address.base; FaultingCodeOffset fco; if (is_intn(offset, 12)) { fco = FaultingCodeOffset(currentOffset()); fld(ft, base, offset); } else { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(base != scratch); ma_li(scratch, Imm32(offset)); ma_add64(scratch, base, scratch); fco = FaultingCodeOffset(currentOffset()); fld(ft, scratch, 0); } return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_fst_d(FloatRegister ft, Address address) { int32_t offset = address.offset; Register base = address.base; FaultingCodeOffset fco; if (is_intn(offset, 12)) { fco = FaultingCodeOffset(currentOffset()); fsd(ft, base, offset); } else { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(base != scratch); ma_li(scratch, Imm32(offset)); ma_add64(scratch, base, scratch); fco = FaultingCodeOffset(currentOffset()); fsd(ft, scratch, 0); } return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_fst_s(FloatRegister ft, Address address) { int32_t offset = address.offset; Register base = address.base; FaultingCodeOffset fco; if (is_intn(offset, 12)) { fco = FaultingCodeOffset(currentOffset()); fsw(ft, base, offset); } else { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); MOZ_ASSERT(base != scratch); ma_li(scratch, Imm32(offset)); ma_add64(scratch, base, scratch); fco = FaultingCodeOffset(currentOffset()); fsw(ft, scratch, 0); } return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_fst_d(FloatRegister ft, BaseIndex address) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); asMasm().computeScaledAddress(address, scratch); FaultingCodeOffset fco = FaultingCodeOffset(currentOffset()); asMasm().ma_fst_d(ft, Address(scratch, address.offset)); return fco; } FaultingCodeOffset MacroAssemblerRiscv64::ma_fst_s(FloatRegister ft, BaseIndex address) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); asMasm().computeScaledAddress(address, scratch); FaultingCodeOffset fco = FaultingCodeOffset(currentOffset()); asMasm().ma_fst_s(ft, Address(scratch, address.offset)); return fco; } void MacroAssemblerRiscv64::ma_fld_d(FloatRegister ft, const BaseIndex& src) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); asMasm().computeScaledAddress(src, scratch); asMasm().ma_fld_d(ft, Address(scratch, src.offset)); } void MacroAssemblerRiscv64::ma_fld_s(FloatRegister ft, const BaseIndex& src) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); asMasm().computeScaledAddress(src, scratch); asMasm().ma_fld_s(ft, Address(scratch, src.offset)); } void MacroAssemblerRiscv64::ma_call(ImmPtr dest) { DEBUG_PRINTF("[ %s\n", __FUNCTION__); BlockTrampolinePoolScope block_trampoline_pool(this, 8); UseScratchRegisterScope temps(this); temps.Exclude(GeneralRegisterSet(1 << CallReg.code())); asMasm().ma_liPatchable(CallReg, dest); jalr(CallReg, 0); DEBUG_PRINTF("]\n"); } void MacroAssemblerRiscv64::CompareIsNotNanF32(Register rd, FPURegister cmp1, FPURegister cmp2) { UseScratchRegisterScope temps(this); BlockTrampolinePoolScope block_trampoline_pool(this, 3); Register scratch = temps.Acquire(); feq_s(rd, cmp1, cmp1); // rd <- !isNan(cmp1) feq_s(scratch, cmp2, cmp2); // scratch <- !isNaN(cmp2) ma_and(rd, rd, scratch); // rd <- !isNan(cmp1) && !isNan(cmp2) } void MacroAssemblerRiscv64::CompareIsNotNanF64(Register rd, FPURegister cmp1, FPURegister cmp2) { UseScratchRegisterScope temps(this); BlockTrampolinePoolScope block_trampoline_pool(this, 3); Register scratch = temps.Acquire(); feq_d(rd, cmp1, cmp1); // rd <- !isNan(cmp1) feq_d(scratch, cmp2, cmp2); // scratch <- !isNaN(cmp2) ma_and(rd, rd, scratch); // rd <- !isNan(cmp1) && !isNan(cmp2) } void MacroAssemblerRiscv64::CompareIsNanF32(Register rd, FPURegister cmp1, FPURegister cmp2) { CompareIsNotNanF32(rd, cmp1, cmp2); // rd <- !isNan(cmp1) && !isNan(cmp2) ma_xor(rd, rd, Operand(1)); // rd <- isNan(cmp1) || isNan(cmp2) } void MacroAssemblerRiscv64::CompareIsNanF64(Register rd, FPURegister cmp1, FPURegister cmp2) { CompareIsNotNanF64(rd, cmp1, cmp2); // rd <- !isNan(cmp1) && !isNan(cmp2) ma_xor(rd, rd, Operand(1)); // rd <- isNan(cmp1) || isNan(cmp2) } void MacroAssemblerRiscv64::Clz32(Register rd, Register xx) { // 32 bit unsigned in lower word: count number of leading zeros. // int n = 32; // unsigned y; // y = x >>16; if (y != 0) { n = n -16; x = y; } // y = x >> 8; if (y != 0) { n = n - 8; x = y; } // y = x >> 4; if (y != 0) { n = n - 4; x = y; } // y = x >> 2; if (y != 0) { n = n - 2; x = y; } // y = x >> 1; if (y != 0) {rd = n - 2; return;} // rd = n - x; Label L0, L1, L2, L3, L4; UseScratchRegisterScope temps(this); Register x = rd; Register y = temps.Acquire(); Register n = temps.Acquire(); MOZ_ASSERT(xx != y && xx != n); mv(x, xx); ma_li(n, Imm32(32)); #if JS_CODEGEN_RISCV64 srliw(y, x, 16); ma_branch(&L0, Equal, y, Operand(zero_reg)); mv(x, y); addiw(n, n, -16); bind(&L0); srliw(y, x, 8); ma_branch(&L1, Equal, y, Operand(zero_reg)); addiw(n, n, -8); mv(x, y); bind(&L1); srliw(y, x, 4); ma_branch(&L2, Equal, y, Operand(zero_reg)); addiw(n, n, -4); mv(x, y); bind(&L2); srliw(y, x, 2); ma_branch(&L3, Equal, y, Operand(zero_reg)); addiw(n, n, -2); mv(x, y); bind(&L3); srliw(y, x, 1); subw(rd, n, x); ma_branch(&L4, Equal, y, Operand(zero_reg)); addiw(rd, n, -2); bind(&L4); #elif JS_CODEGEN_RISCV32 srli(y, x, 16); ma_branch(&L0, Equal, y, Operand(zero_reg)); mv(x, y); addi(n, n, -16); bind(&L0); srli(y, x, 8); ma_branch(&L1, Equal, y, Operand(zero_reg)); addi(n, n, -8); mv(x, y); bind(&L1); srli(y, x, 4); ma_branch(&L2, Equal, y, Operand(zero_reg)); addi(n, n, -4); mv(x, y); bind(&L2); srli(y, x, 2); ma_branch(&L3, Equal, y, Operand(zero_reg)); addi(n, n, -2); mv(x, y); bind(&L3); srli(y, x, 1); sub(rd, n, x); ma_branch(&L4, Equal, y, Operand(zero_reg)); addi(rd, n, -2); bind(&L4); #endif } #if JS_CODEGEN_RISCV64 void MacroAssemblerRiscv64::Clz64(Register rd, Register xx) { // 64 bit: count number of leading zeros. // int n = 64; // unsigned y; // y = x >>32; if (y != 0) { n = n - 32; x = y; } // y = x >>16; if (y != 0) { n = n - 16; x = y; } // y = x >> 8; if (y != 0) { n = n - 8; x = y; } // y = x >> 4; if (y != 0) { n = n - 4; x = y; } // y = x >> 2; if (y != 0) { n = n - 2; x = y; } // y = x >> 1; if (y != 0) {rd = n - 2; return;} // rd = n - x; Label L0, L1, L2, L3, L4, L5; UseScratchRegisterScope temps(this); Register x = rd; Register y = temps.Acquire(); Register n = temps.Acquire(); MOZ_ASSERT(xx != y && xx != n); mv(x, xx); ma_li(n, Imm32(64)); srli(y, x, 32); ma_branch(&L0, Equal, y, Operand(zero_reg)); addiw(n, n, -32); mv(x, y); bind(&L0); srli(y, x, 16); ma_branch(&L1, Equal, y, Operand(zero_reg)); addiw(n, n, -16); mv(x, y); bind(&L1); srli(y, x, 8); ma_branch(&L2, Equal, y, Operand(zero_reg)); addiw(n, n, -8); mv(x, y); bind(&L2); srli(y, x, 4); ma_branch(&L3, Equal, y, Operand(zero_reg)); addiw(n, n, -4); mv(x, y); bind(&L3); srli(y, x, 2); ma_branch(&L4, Equal, y, Operand(zero_reg)); addiw(n, n, -2); mv(x, y); bind(&L4); srli(y, x, 1); subw(rd, n, x); ma_branch(&L5, Equal, y, Operand(zero_reg)); addiw(rd, n, -2); bind(&L5); } #endif void MacroAssemblerRiscv64::Ctz32(Register rd, Register rs) { // Convert trailing zeroes to trailing ones, and bits to their left // to zeroes. { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_add64(scratch, rs, Operand(-1)); ma_xor(rd, scratch, rs); ma_and(rd, rd, scratch); // Count number of leading zeroes. } Clz32(rd, rd); { // Subtract number of leading zeroes from 32 to get number of trailing // ones. Remember that the trailing ones were formerly trailing zeroes. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, Imm32(32)); ma_sub32(rd, scratch, rd); } } #if JS_CODEGEN_RISCV64 void MacroAssemblerRiscv64::Ctz64(Register rd, Register rs) { // Convert trailing zeroes to trailing ones, and bits to their left // to zeroes. { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_add64(scratch, rs, Operand(-1)); ma_xor(rd, scratch, rs); ma_and(rd, rd, scratch); // Count number of leading zeroes. } Clz64(rd, rd); { // Subtract number of leading zeroes from 64 to get number of trailing // ones. Remember that the trailing ones were formerly trailing zeroes. UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, 64); ma_sub64(rd, scratch, rd); } } #endif void MacroAssemblerRiscv64::Popcnt32(Register rd, Register rs, Register scratch) { MOZ_ASSERT(scratch != rs); MOZ_ASSERT(scratch != rd); // https://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetParallel // // A generalization of the best bit counting method to integers of // bit-widths up to 128 (parameterized by type T) is this: // // v = v - ((v >> 1) & (T)~(T)0/3); // temp // v = (v & (T)~(T)0/15*3) + ((v >> 2) & (T)~(T)0/15*3); // temp // v = (v + (v >> 4)) & (T)~(T)0/255*15; // temp // c = (T)(v * ((T)~(T)0/255)) >> (sizeof(T) - 1) * BITS_PER_BYTE; //count // // There are algorithms which are faster in the cases where very few // bits are set but the algorithm here attempts to minimize the total // number of instructions executed even when a large number of bits // are set. // The number of instruction is 20. // uint32_t B0 = 0x55555555; // (T)~(T)0/3 // uint32_t B1 = 0x33333333; // (T)~(T)0/15*3 // uint32_t B2 = 0x0F0F0F0F; // (T)~(T)0/255*15 // uint32_t value = 0x01010101; // (T)~(T)0/255 uint32_t shift = 24; UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); Register value = temps.Acquire(); MOZ_ASSERT((rd != value) && (rs != value)); ma_li(value, 0x01010101); // value = 0x01010101; ma_li(scratch2, 0x55555555); // B0 = 0x55555555; ma_srl32(scratch, rs, Operand(1)); ma_and(scratch, scratch, scratch2); ma_sub32(scratch, rs, scratch); ma_li(scratch2, 0x33333333); // B1 = 0x33333333; slli(rd, scratch2, 4); or_(scratch2, scratch2, rd); ma_and(rd, scratch, scratch2); ma_srl32(scratch, scratch, Operand(2)); ma_and(scratch, scratch, scratch2); ma_add32(scratch, rd, scratch); ma_srl32(rd, scratch, Operand(4)); ma_add32(rd, rd, scratch); ma_li(scratch2, 0xF); ma_mul32(scratch2, value, scratch2); // B2 = 0x0F0F0F0F; ma_and(rd, rd, scratch2); ma_mul32(rd, rd, value); ma_srl32(rd, rd, Operand(shift)); } #if JS_CODEGEN_RISCV64 void MacroAssemblerRiscv64::Popcnt64(Register rd, Register rs, Register scratch) { MOZ_ASSERT(scratch != rs); MOZ_ASSERT(scratch != rd); // uint64_t B0 = 0x5555555555555555l; // (T)~(T)0/3 // uint64_t B1 = 0x3333333333333333l; // (T)~(T)0/15*3 // uint64_t B2 = 0x0F0F0F0F0F0F0F0Fl; // (T)~(T)0/255*15 // uint64_t value = 0x0101010101010101l; // (T)~(T)0/255 // uint64_t shift = 24; // (sizeof(T) - 1) * BITS_PER_BYTE uint64_t shift = 24; UseScratchRegisterScope temps(this); Register scratch2 = temps.Acquire(); Register value = temps.Acquire(); MOZ_ASSERT((rd != value) && (rs != value)); ma_li(value, 0x1111111111111111l); // value = 0x1111111111111111l; ma_li(scratch2, 5); ma_mul64(scratch2, value, scratch2); // B0 = 0x5555555555555555l; ma_srl64(scratch, rs, Operand(1)); ma_and(scratch, scratch, scratch2); ma_sub64(scratch, rs, scratch); ma_li(scratch2, 3); ma_mul64(scratch2, value, scratch2); // B1 = 0x3333333333333333l; ma_and(rd, scratch, scratch2); ma_srl64(scratch, scratch, Operand(2)); ma_and(scratch, scratch, scratch2); ma_add64(scratch, rd, scratch); ma_srl64(rd, scratch, Operand(4)); ma_add64(rd, rd, scratch); ma_li(scratch2, 0xF); ma_li(value, 0x0101010101010101l); // value = 0x0101010101010101l; ma_mul64(scratch2, value, scratch2); // B2 = 0x0F0F0F0F0F0F0F0Fl; ma_and(rd, rd, scratch2); ma_mul64(rd, rd, value); srli(rd, rd, 32 + shift); } #endif void MacroAssemblerRiscv64::ma_div_branch_overflow(Register rd, Register rj, Register rk, Label* overflow) { ScratchRegisterScope scratch(asMasm()); ma_mod32(scratch, rj, rk); ma_b(scratch, scratch, overflow, Assembler::NonZero); divw(rd, rj, rk); } void MacroAssemblerRiscv64::ma_div_branch_overflow(Register rd, Register rj, Imm32 imm, Label* overflow) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); ma_li(scratch, imm); ma_div_branch_overflow(rd, rj, scratch, overflow); } void MacroAssemblerRiscv64::ma_mod_mask(Register src, Register dest, Register hold, Register remain, int32_t shift, Label* negZero) { // MATH: // We wish to compute x % (1< 0, store sum - C back into sum, thus performing a // modulus. ma_b(scratch2, Register(scratch2), &sumSigned, Signed, ShortJump); or_(dest, scratch2, zero); bind(&sumSigned); // Get rid of the bits that we extracted before. srliw(remain, remain, shift); // If the shift produced zero, finish, otherwise, continue in the loop. ma_b(remain, remain, &head, NonZero, ShortJump); // Check the hold to see if we need to negate the result. ma_b(hold, hold, &done, NotSigned, ShortJump); // If the hold was non-zero, negate the result to be in line with // what JS wants if (negZero != nullptr) { // Jump out in case of negative zero. ma_b(hold, hold, negZero, Zero); subw(dest, zero, dest); } else { subw(dest, zero, dest); } bind(&done); } void MacroAssemblerRiscv64::ma_fmovz(FloatFormat fmt, FloatRegister fd, FloatRegister fj, Register rk) { Label done; ma_b(rk, zero, &done, Assembler::NotEqual); if (fmt == SingleFloat) { fmv_s(fd, fj); } else { fmv_d(fd, fj); } bind(&done); } void MacroAssemblerRiscv64::ByteSwap(Register rd, Register rs, int operand_size, Register scratch) { MOZ_ASSERT(scratch != rs); MOZ_ASSERT(scratch != rd); MOZ_ASSERT(operand_size == 4 || operand_size == 8); if (operand_size == 4) { // Uint32_t x1 = 0x00FF00FF; // x0 = (x0 << 16 | x0 >> 16); // x0 = (((x0 & x1) << 8) | ((x0 & (x1 << 8)) >> 8)); UseScratchRegisterScope temps(this); BlockTrampolinePoolScope block_trampoline_pool(this, 17); MOZ_ASSERT((rd != t6) && (rs != t6)); Register x0 = temps.Acquire(); Register x1 = temps.Acquire(); Register x2 = scratch; RV_li(x1, 0x00FF00FF); slliw(x0, rs, 16); srliw(rd, rs, 16); or_(x0, rd, x0); // x0 <- x0 << 16 | x0 >> 16 and_(x2, x0, x1); // x2 <- x0 & 0x00FF00FF slliw(x2, x2, 8); // x2 <- (x0 & x1) << 8 slliw(x1, x1, 8); // x1 <- 0xFF00FF00 and_(rd, x0, x1); // x0 & 0xFF00FF00 srliw(rd, rd, 8); or_(rd, rd, x2); // (((x0 & x1) << 8) | ((x0 & (x1 << 8)) >> 8)) } else { // uinx24_t x1 = 0x0000FFFF0000FFFFl; // uinx24_t x1 = 0x00FF00FF00FF00FFl; // x0 = (x0 << 32 | x0 >> 32); // x0 = (x0 & x1) << 16 | (x0 & (x1 << 16)) >> 16; // x0 = (x0 & x1) << 8 | (x0 & (x1 << 8)) >> 8; UseScratchRegisterScope temps(this); BlockTrampolinePoolScope block_trampoline_pool(this, 30); MOZ_ASSERT((rd != t6) && (rs != t6)); Register x0 = temps.Acquire(); Register x1 = temps.Acquire(); Register x2 = scratch; RV_li(x1, 0x0000FFFF0000FFFFl); slli(x0, rs, 32); srli(rd, rs, 32); or_(x0, rd, x0); // x0 <- x0 << 32 | x0 >> 32 and_(x2, x0, x1); // x2 <- x0 & 0x0000FFFF0000FFFF slli(x2, x2, 16); // x2 <- (x0 & 0x0000FFFF0000FFFF) << 16 slli(x1, x1, 16); // x1 <- 0xFFFF0000FFFF0000 and_(rd, x0, x1); // rd <- x0 & 0xFFFF0000FFFF0000 srli(rd, rd, 16); // rd <- x0 & (x1 << 16)) >> 16 or_(x0, rd, x2); // (x0 & x1) << 16 | (x0 & (x1 << 16)) >> 16; RV_li(x1, 0x00FF00FF00FF00FFl); and_(x2, x0, x1); // x2 <- x0 & 0x00FF00FF00FF00FF slli(x2, x2, 8); // x2 <- (x0 & x1) << 8 slli(x1, x1, 8); // x1 <- 0xFF00FF00FF00FF00 and_(rd, x0, x1); srli(rd, rd, 8); // rd <- (x0 & (x1 << 8)) >> 8 or_(rd, rd, x2); // (((x0 & x1) << 8) | ((x0 & (x1 << 8)) >> 8)) } } template void MacroAssemblerRiscv64::FloatMinMaxHelper(FPURegister dst, FPURegister src1, FPURegister src2, MaxMinKind kind) { MOZ_ASSERT((std::is_same::value) || (std::is_same::value)); if (src1 == src2 && dst != src1) { if (std::is_same::value) { fmv_s(dst, src1); } else { fmv_d(dst, src1); } return; } Label done, nan; // For RISCV, fmin_s returns the other non-NaN operand as result if only one // operand is NaN; but for JS, if any operand is NaN, result is Nan. The // following handles the discrepency between handling of NaN between ISA and // JS semantics UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); if (std::is_same::value) { CompareIsNotNanF32(scratch, src1, src2); } else { CompareIsNotNanF64(scratch, src1, src2); } BranchFalseF(scratch, &nan); if (kind == MaxMinKind::kMax) { if (std::is_same::value) { fmax_s(dst, src1, src2); } else { fmax_d(dst, src1, src2); } } else { if (std::is_same::value) { fmin_s(dst, src1, src2); } else { fmin_d(dst, src1, src2); } } jump(&done); bind(&nan); // if any operand is NaN, return NaN (fadd returns NaN if any operand is NaN) if (std::is_same::value) { fadd_s(dst, src1, src2); } else { fadd_d(dst, src1, src2); } bind(&done); } void MacroAssemblerRiscv64::Float32Max(FPURegister dst, FPURegister src1, FPURegister src2) { comment(__FUNCTION__); FloatMinMaxHelper(dst, src1, src2, MaxMinKind::kMax); } void MacroAssemblerRiscv64::Float32Min(FPURegister dst, FPURegister src1, FPURegister src2) { comment(__FUNCTION__); FloatMinMaxHelper(dst, src1, src2, MaxMinKind::kMin); } void MacroAssemblerRiscv64::Float64Max(FPURegister dst, FPURegister src1, FPURegister src2) { comment(__FUNCTION__); FloatMinMaxHelper(dst, src1, src2, MaxMinKind::kMax); } void MacroAssemblerRiscv64::Float64Min(FPURegister dst, FPURegister src1, FPURegister src2) { comment(__FUNCTION__); FloatMinMaxHelper(dst, src1, src2, MaxMinKind::kMin); } void MacroAssemblerRiscv64::BranchTrueShortF(Register rs, Label* target) { ma_branch(target, NotEqual, rs, Operand(zero_reg)); } void MacroAssemblerRiscv64::BranchFalseShortF(Register rs, Label* target) { ma_branch(target, Equal, rs, Operand(zero_reg)); } void MacroAssemblerRiscv64::BranchTrueF(Register rs, Label* target) { bool long_branch = target->bound() ? !is_near(target) : false; if (long_branch) { Label skip; BranchFalseShortF(rs, &skip); BranchLong(target); bind(&skip); } else { BranchTrueShortF(rs, target); } } void MacroAssemblerRiscv64::BranchFalseF(Register rs, Label* target) { bool long_branch = target->bound() ? !is_near(target) : false; if (long_branch) { Label skip; BranchTrueShortF(rs, &skip); BranchLong(target); bind(&skip); } else { BranchFalseShortF(rs, target); } } void MacroAssemblerRiscv64::Ror(Register rd, Register rs, const Operand& rt) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 8); if (rt.is_reg()) { negw(scratch, rt.rm()); sllw(scratch, rs, scratch); srlw(rd, rs, rt.rm()); or_(rd, scratch, rd); sext_w(rd, rd); } else { int64_t ror_value = rt.immediate() % 32; if (ror_value == 0) { mv(rd, rs); return; } else if (ror_value < 0) { ror_value += 32; } srliw(scratch, rs, ror_value); slliw(rd, rs, 32 - ror_value); or_(rd, scratch, rd); sext_w(rd, rd); } } void MacroAssemblerRiscv64::Dror(Register rd, Register rs, const Operand& rt) { UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); BlockTrampolinePoolScope block_trampoline_pool(this, 8); if (rt.is_reg()) { negw(scratch, rt.rm()); sll(scratch, rs, scratch); srl(rd, rs, rt.rm()); or_(rd, scratch, rd); } else { int64_t dror_value = rt.immediate() % 64; if (dror_value == 0) { mv(rd, rs); return; } else if (dror_value < 0) { dror_value += 64; } srli(scratch, rs, dror_value); slli(rd, rs, 64 - dror_value); or_(rd, scratch, rd); } } void MacroAssemblerRiscv64::wasmLoadImpl(const wasm::MemoryAccessDesc& access, Register memoryBase, Register ptr, Register ptrScratch, AnyRegister output, Register tmp) { access.assertOffsetInGuardPages(); uint32_t offset = access.offset(); MOZ_ASSERT_IF(offset, ptrScratch != InvalidReg); // Maybe add the offset. if (offset) { asMasm().addPtr(ImmWord(offset), ptrScratch); ptr = ptrScratch; } asMasm().memoryBarrierBefore(access.sync()); UseScratchRegisterScope temps(this); Register scratch = temps.Acquire(); FaultingCodeOffset fco; switch (access.type()) { case Scalar::Int8: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lb(output.gpr(), scratch, 0); break; case Scalar::Uint8: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lbu(output.gpr(), scratch, 0); break; case Scalar::Int16: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lh(output.gpr(), scratch, 0); break; case Scalar::Uint16: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lhu(output.gpr(), scratch, 0); break; case Scalar::Int32: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lw(output.gpr(), scratch, 0); break; case Scalar::Uint32: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); lwu(output.gpr(), scratch, 0); break; case Scalar::Float64: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); fld(output.fpu(), scratch, 0); break; case Scalar::Float32: add(scratch, memoryBase, ptr); fco = FaultingCodeOffset(currentOffset()); flw(output.fpu(), scratch, 0); break; default: MOZ_CRASH("unexpected array type"); } asMasm().append(access, js::wasm::TrapMachineInsnForLoad(access.byteSize()), fco); asMasm().memoryBarrierAfter(access.sync()); } void MacroAssemblerRiscv64::wasmStoreImpl(const wasm::MemoryAccessDesc& access, AnyRegister value, Register memoryBase, Register ptr, Register ptrScratch, Register tmp) { access.assertOffsetInGuardPages(); uint32_t offset = access.offset(); MOZ_ASSERT_IF(offset, ptrScratch != InvalidReg); // Maybe add the offset. if (offset) { asMasm().addPtr(ImmWord(offset), ptrScratch); ptr = ptrScratch; } unsigned byteSize = access.byteSize(); bool isSigned; bool isFloat = false; switch (access.type()) { case Scalar::Int8: isSigned = true; break; case Scalar::Uint8: isSigned = false; break; case Scalar::Int16: isSigned = true; break; case Scalar::Uint16: isSigned = false; break; case Scalar::Int32: isSigned = true; break; case Scalar::Uint32: isSigned = false; break; case Scalar::Int64: isSigned = true; break; case Scalar::Float64: isFloat = true; break; case Scalar::Float32: isFloat = true; break; default: MOZ_CRASH("unexpected array type"); } BaseIndex address(memoryBase, ptr, TimesOne); asMasm().memoryBarrierBefore(access.sync()); FaultingCodeOffset fco; if (isFloat) { if (byteSize == 4) { fco = asMasm().ma_fst_s(value.fpu(), address); } else { fco = asMasm().ma_fst_d(value.fpu(), address); } } else { fco = asMasm().ma_store(value.gpr(), address, static_cast(8 * byteSize), isSigned ? SignExtend : ZeroExtend); } // Only the last emitted instruction is a memory access. asMasm().append(access, js::wasm::TrapMachineInsnForStore(access.byteSize()), fco); asMasm().memoryBarrierAfter(access.sync()); } void MacroAssemblerRiscv64::GenPCRelativeJumpAndLink(Register rd, int32_t imm32) { MOZ_ASSERT(is_int32(imm32 + 0x800)); int32_t Hi20 = ((imm32 + 0x800) >> 12); int32_t Lo12 = imm32 << 20 >> 20; auipc(rd, Hi20); // Read PC + Hi20 into scratch. jalr(rd, Lo12); // jump PC + Hi20 + Lo12 } void MacroAssemblerRiscv64::BranchAndLinkShortHelper(int32_t offset, Label* L) { MOZ_ASSERT(L == nullptr || offset == 0); offset = GetOffset(offset, L, OffsetSize::kOffset21); jal(offset); } void MacroAssemblerRiscv64::BranchAndLinkShort(int32_t offset) { MOZ_ASSERT(is_int21(offset)); BranchAndLinkShortHelper(offset, nullptr); } void MacroAssemblerRiscv64::BranchAndLinkShort(Label* L) { BranchAndLinkShortHelper(0, L); } void MacroAssemblerRiscv64::BranchAndLink(Label* L) { if (L->bound()) { if (is_near(L)) { BranchAndLinkShort(L); } else { BranchAndLinkLong(L); } } else { BranchAndLinkShort(L); } } void MacroAssemblerRiscv64::ma_fmv_d(FloatRegister src, ValueOperand dest) { fmv_x_d(dest.valueReg(), src); } void MacroAssemblerRiscv64::ma_fmv_d(ValueOperand src, FloatRegister dest) { fmv_d_x(dest, src.valueReg()); } void MacroAssemblerRiscv64::ma_fmv_w(FloatRegister src, ValueOperand dest) { fmv_x_w(dest.valueReg(), src); } void MacroAssemblerRiscv64::ma_fmv_w(ValueOperand src, FloatRegister dest) { fmv_w_x(dest, src.valueReg()); } } // namespace jit } // namespace js