Add support for i32x4_trunc_sat_f64x2_u for x64
This commit is contained in:
@@ -631,6 +631,7 @@ pub enum SseOpcode {
|
|||||||
Roundss,
|
Roundss,
|
||||||
Roundsd,
|
Roundsd,
|
||||||
Rsqrtss,
|
Rsqrtss,
|
||||||
|
Shufps,
|
||||||
Sqrtps,
|
Sqrtps,
|
||||||
Sqrtpd,
|
Sqrtpd,
|
||||||
Sqrtss,
|
Sqrtss,
|
||||||
@@ -677,6 +678,7 @@ impl SseOpcode {
|
|||||||
| SseOpcode::Orps
|
| SseOpcode::Orps
|
||||||
| SseOpcode::Rcpss
|
| SseOpcode::Rcpss
|
||||||
| SseOpcode::Rsqrtss
|
| SseOpcode::Rsqrtss
|
||||||
|
| SseOpcode::Shufps
|
||||||
| SseOpcode::Sqrtps
|
| SseOpcode::Sqrtps
|
||||||
| SseOpcode::Sqrtss
|
| SseOpcode::Sqrtss
|
||||||
| SseOpcode::Subps
|
| SseOpcode::Subps
|
||||||
@@ -1003,6 +1005,7 @@ impl fmt::Debug for SseOpcode {
|
|||||||
SseOpcode::Roundss => "roundss",
|
SseOpcode::Roundss => "roundss",
|
||||||
SseOpcode::Roundsd => "roundsd",
|
SseOpcode::Roundsd => "roundsd",
|
||||||
SseOpcode::Rsqrtss => "rsqrtss",
|
SseOpcode::Rsqrtss => "rsqrtss",
|
||||||
|
SseOpcode::Shufps => "shufps",
|
||||||
SseOpcode::Sqrtps => "sqrtps",
|
SseOpcode::Sqrtps => "sqrtps",
|
||||||
SseOpcode::Sqrtpd => "sqrtpd",
|
SseOpcode::Sqrtpd => "sqrtpd",
|
||||||
SseOpcode::Sqrtss => "sqrtss",
|
SseOpcode::Sqrtss => "sqrtss",
|
||||||
|
|||||||
@@ -1699,6 +1699,7 @@ pub(crate) fn emit(
|
|||||||
SseOpcode::Roundss => (LegacyPrefixes::_66, 0x0F3A0A, 3),
|
SseOpcode::Roundss => (LegacyPrefixes::_66, 0x0F3A0A, 3),
|
||||||
SseOpcode::Roundpd => (LegacyPrefixes::_66, 0x0F3A09, 3),
|
SseOpcode::Roundpd => (LegacyPrefixes::_66, 0x0F3A09, 3),
|
||||||
SseOpcode::Roundsd => (LegacyPrefixes::_66, 0x0F3A0B, 3),
|
SseOpcode::Roundsd => (LegacyPrefixes::_66, 0x0F3A0B, 3),
|
||||||
|
SseOpcode::Shufps => (LegacyPrefixes::None, 0x0FC6, 2),
|
||||||
_ => unimplemented!("Opcode {:?} not implemented", op),
|
_ => unimplemented!("Opcode {:?} not implemented", op),
|
||||||
};
|
};
|
||||||
let rex = RexFlags::from(*size);
|
let rex = RexFlags::from(*size);
|
||||||
|
|||||||
@@ -4125,6 +4125,18 @@ fn test_x64_emit() {
|
|||||||
"palignr $3, %xmm1, %xmm9",
|
"palignr $3, %xmm1, %xmm9",
|
||||||
));
|
));
|
||||||
|
|
||||||
|
insns.push((
|
||||||
|
Inst::xmm_rm_r_imm(
|
||||||
|
SseOpcode::Shufps,
|
||||||
|
RegMem::reg(xmm1),
|
||||||
|
w_xmm10,
|
||||||
|
136,
|
||||||
|
OperandSize::Size32,
|
||||||
|
),
|
||||||
|
"440FC6D188",
|
||||||
|
"shufps $136, %xmm1, %xmm10",
|
||||||
|
));
|
||||||
|
|
||||||
insns.push((
|
insns.push((
|
||||||
Inst::xmm_rm_r_imm(
|
Inst::xmm_rm_r_imm(
|
||||||
SseOpcode::Roundps,
|
SseOpcode::Roundps,
|
||||||
|
|||||||
@@ -6442,6 +6442,84 @@ fn lower_insn_to_regs<C: LowerCtx<I = Inst>>(
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Opcode::Uunarrow => {
|
||||||
|
if let Some(fcvt_inst) = matches_input(ctx, inputs[0], Opcode::FcvtToUintSat) {
|
||||||
|
//y = i32x4.trunc_sat_f64x2_u_zero(x) is lowered to:
|
||||||
|
//MOVAPD xmm_y, xmm_x
|
||||||
|
//XORPD xmm_tmp, xmm_tmp
|
||||||
|
//MAXPD xmm_y, xmm_tmp
|
||||||
|
//MINPD xmm_y, [wasm_f64x2_splat(4294967295.0)]
|
||||||
|
//ROUNDPD xmm_y, xmm_y, 0x0B
|
||||||
|
//ADDPD xmm_y, [wasm_f64x2_splat(0x1.0p+52)]
|
||||||
|
//SHUFPS xmm_y, xmm_xmp, 0x88
|
||||||
|
|
||||||
|
let fcvt_input = InsnInput {
|
||||||
|
insn: fcvt_inst,
|
||||||
|
input: 0,
|
||||||
|
};
|
||||||
|
let input_ty = ctx.input_ty(fcvt_inst, 0);
|
||||||
|
let output_ty = ctx.output_ty(insn, 0);
|
||||||
|
let src = put_input_in_reg(ctx, fcvt_input);
|
||||||
|
let dst = get_output_reg(ctx, outputs[0]).only_reg().unwrap();
|
||||||
|
|
||||||
|
ctx.emit(Inst::gen_move(dst, src, input_ty));
|
||||||
|
let tmp1 = ctx.alloc_tmp(output_ty).only_reg().unwrap();
|
||||||
|
ctx.emit(Inst::xmm_rm_r(SseOpcode::Xorpd, RegMem::from(tmp1), tmp1));
|
||||||
|
ctx.emit(Inst::xmm_rm_r(SseOpcode::Maxpd, RegMem::from(tmp1), dst));
|
||||||
|
|
||||||
|
// 4294967295.0 is equivalent to 0x41EFFFFFFFE00000
|
||||||
|
static UMAX_MASK: [u8; 16] = [
|
||||||
|
0x00, 0x00, 0xE0, 0xFF, 0xFF, 0xFF, 0xEF, 0x41, 0x00, 0x00, 0xE0, 0xFF, 0xFF,
|
||||||
|
0xFF, 0xEF, 0x41,
|
||||||
|
];
|
||||||
|
let umax_const = ctx.use_constant(VCodeConstantData::WellKnown(&UMAX_MASK));
|
||||||
|
let umax_mask = ctx.alloc_tmp(types::F64X2).only_reg().unwrap();
|
||||||
|
ctx.emit(Inst::xmm_load_const(umax_const, umax_mask, types::F64X2));
|
||||||
|
|
||||||
|
//MINPD xmm_y, [wasm_f64x2_splat(4294967295.0)]
|
||||||
|
ctx.emit(Inst::xmm_rm_r(
|
||||||
|
SseOpcode::Minpd,
|
||||||
|
RegMem::from(umax_mask),
|
||||||
|
dst,
|
||||||
|
));
|
||||||
|
//ROUNDPD xmm_y, xmm_y, 0x0B
|
||||||
|
ctx.emit(Inst::xmm_rm_r_imm(
|
||||||
|
SseOpcode::Roundpd,
|
||||||
|
RegMem::reg(dst.to_reg()),
|
||||||
|
dst,
|
||||||
|
RoundImm::RoundZero.encode(),
|
||||||
|
OperandSize::Size32,
|
||||||
|
));
|
||||||
|
//ADDPD xmm_y, [wasm_f64x2_splat(0x1.0p+52)]
|
||||||
|
static UINT_MASK: [u8; 16] = [
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x30, 0x43, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x30, 0x43,
|
||||||
|
];
|
||||||
|
let uint_mask_const = ctx.use_constant(VCodeConstantData::WellKnown(&UINT_MASK));
|
||||||
|
let uint_mask = ctx.alloc_tmp(types::F64X2).only_reg().unwrap();
|
||||||
|
ctx.emit(Inst::xmm_load_const(
|
||||||
|
uint_mask_const,
|
||||||
|
uint_mask,
|
||||||
|
types::F64X2,
|
||||||
|
));
|
||||||
|
ctx.emit(Inst::xmm_rm_r(
|
||||||
|
SseOpcode::Addpd,
|
||||||
|
RegMem::from(uint_mask),
|
||||||
|
dst,
|
||||||
|
));
|
||||||
|
|
||||||
|
//SHUFPS xmm_y, xmm_xmp, 0x88
|
||||||
|
ctx.emit(Inst::xmm_rm_r_imm(
|
||||||
|
SseOpcode::Shufps,
|
||||||
|
RegMem::reg(tmp1.to_reg()),
|
||||||
|
dst,
|
||||||
|
0x88,
|
||||||
|
OperandSize::Size32,
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
println!("Did not match fcvt input!");
|
||||||
|
}
|
||||||
|
}
|
||||||
// Unimplemented opcodes below. These are not currently used by Wasm
|
// Unimplemented opcodes below. These are not currently used by Wasm
|
||||||
// lowering or other known embeddings, but should be either supported or
|
// lowering or other known embeddings, but should be either supported or
|
||||||
// removed eventually.
|
// removed eventually.
|
||||||
@@ -6472,10 +6550,6 @@ fn lower_insn_to_regs<C: LowerCtx<I = Inst>>(
|
|||||||
unimplemented!("Vector split/concat ops not implemented.");
|
unimplemented!("Vector split/concat ops not implemented.");
|
||||||
}
|
}
|
||||||
|
|
||||||
Opcode::Uunarrow => {
|
|
||||||
unimplemented!("unimplemented lowering for opcode {:?}", op);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Opcodes that should be removed by legalization. These should
|
// Opcodes that should be removed by legalization. These should
|
||||||
// eventually be removed if/when we replace in-situ legalization with
|
// eventually be removed if/when we replace in-situ legalization with
|
||||||
// something better.
|
// something better.
|
||||||
|
|||||||
Reference in New Issue
Block a user