alexcrichton labeled issue #5962:
This came up during https://github.com/bytecodealliance/wasmtime/pull/5918 and the removal of the old
vselect
instruction. Currently thebitselect
instruction uses the same type for all of its inputs and its output, but that means that for a floating-point selection the condition mask is also a floating-point value which is typically not the case. For example anicmp
would produce an integer vector which could be used tobitselect
the results of the comparisons.Using an integral mask would make the NaN-canonicalization for vectors slightly simpler and additionally make it easier to re-add the optimizations removed in https://github.com/bytecodealliance/wasmtime/pull/5918 where
bitselect
-of-fcmp
for float vectors could be optimized intof{min,max}
where possible.
alexcrichton opened issue #5962:
This came up during https://github.com/bytecodealliance/wasmtime/pull/5918 and the removal of the old
vselect
instruction. Currently thebitselect
instruction uses the same type for all of its inputs and its output, but that means that for a floating-point selection the condition mask is also a floating-point value which is typically not the case. For example anicmp
would produce an integer vector which could be used tobitselect
the results of the comparisons.Using an integral mask would make the NaN-canonicalization for vectors slightly simpler and additionally make it easier to re-add the optimizations removed in https://github.com/bytecodealliance/wasmtime/pull/5918 where
bitselect
-of-fcmp
for float vectors could be optimized intof{min,max}
where possible.
alexcrichton commented on issue #5962:
I attempted this naively with:
diff --git a/cranelift/codegen/meta/src/shared/instructions.rs b/cranelift/codegen/meta/src/shared/instructions.rs index 372407965..136e8f1bd 100755 --- a/cranelift/codegen/meta/src/shared/instructions.rs +++ b/cranelift/codegen/meta/src/shared/instructions.rs @@ -1449,7 +1449,15 @@ pub(crate) fn define( .side_effects_idempotent(), ); - let c = &Operand::new("c", Any).with_doc("Controlling value to test"); + let BitselectTruthy = &TypeVar::new( + "BitselectTruthy", + "A truthy type for the `bitselect` instruction", + TypeSetBuilder::new() + .ints(Interval::All) + .simd_lanes(Interval::All) + .build(), + ); + let c = &Operand::new("c", BitselectTruthy).with_doc("Controlling value to test"); ig.push( Inst::new( "bitselect", diff --git a/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif b/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif index b4449a967..de531b1e8 100644 --- a/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif +++ b/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif @@ -187,8 +187,8 @@ block0(v0: i16x8, v1: i16x8, v2: i16x8): ; bsl v0.16b, v1.16b, v2.16b ; ret -function %vselect_f32x4(f32x4, f32x4, f32x4) -> f32x4 { -block0(v0: f32x4, v1: f32x4, v2: f32x4): +function %vselect_f32x4(i32x4, f32x4, f32x4) -> f32x4 { +block0(v0: i32x4, v1: f32x4, v2: f32x4): v3 = bitselect v0, v1, v2 return v3 } @@ -203,8 +203,8 @@ block0(v0: f32x4, v1: f32x4, v2: f32x4): ; bsl v0.16b, v1.16b, v2.16b ; ret -function %vselect_f64x2(f64x2, f64x2, f64x2) -> f64x2 { -block0(v0: f64x2, v1: f64x2, v2: f64x2): +function %vselect_f64x2(i64x2, f64x2, f64x2) -> f64x2 { +block0(v0: i64x2, v1: f64x2, v2: f64x2): v3 = bitselect v0, v1, v2 return v3 }
but that's not quite what we want I think since it's allowing
i32
as a mask type forf32x4
bitselects. I don't know how to require that the type size of the condition equals the type size of the mask
alexcrichton edited a comment on issue #5962:
I attempted this naively with:
diff --git a/cranelift/codegen/meta/src/shared/instructions.rs b/cranelift/codegen/meta/src/shared/instructions.rs index 372407965..136e8f1bd 100755 --- a/cranelift/codegen/meta/src/shared/instructions.rs +++ b/cranelift/codegen/meta/src/shared/instructions.rs @@ -1449,7 +1449,15 @@ pub(crate) fn define( .side_effects_idempotent(), ); - let c = &Operand::new("c", Any).with_doc("Controlling value to test"); + let BitselectTruthy = &TypeVar::new( + "BitselectTruthy", + "A truthy type for the `bitselect` instruction", + TypeSetBuilder::new() + .ints(Interval::All) + .simd_lanes(Interval::All) + .build(), + ); + let c = &Operand::new("c", BitselectTruthy).with_doc("Controlling value to test"); ig.push( Inst::new( "bitselect", diff --git a/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif b/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif index b4449a967..de531b1e8 100644 --- a/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif +++ b/cranelift/filetests/filetests/isa/aarch64/simd-bitwise-compile.clif @@ -187,8 +187,8 @@ block0(v0: i16x8, v1: i16x8, v2: i16x8): ; bsl v0.16b, v1.16b, v2.16b ; ret -function %vselect_f32x4(f32x4, f32x4, f32x4) -> f32x4 { -block0(v0: f32x4, v1: f32x4, v2: f32x4): +function %vselect_f32x4(i32x4, f32x4, f32x4) -> f32x4 { +block0(v0: i32x4, v1: f32x4, v2: f32x4): v3 = bitselect v0, v1, v2 return v3 } @@ -203,8 +203,8 @@ block0(v0: f32x4, v1: f32x4, v2: f32x4): ; bsl v0.16b, v1.16b, v2.16b ; ret -function %vselect_f64x2(f64x2, f64x2, f64x2) -> f64x2 { -block0(v0: f64x2, v1: f64x2, v2: f64x2): +function %vselect_f64x2(i64x2, f64x2, f64x2) -> f64x2 { +block0(v0: i64x2, v1: f64x2, v2: f64x2): v3 = bitselect v0, v1, v2 return v3 }
but that's not quite what we want I think since it's allowing
i32
as a mask type forf32x4
bitselects. I don't know how to require that the type size of the condition equals the type size of the oeprands
Last updated: Jan 24 2025 at 00:11 UTC