]> git.lizzy.rs Git - rust.git/commitdiff
Emulate compare simd intrinsics
authorbjorn3 <bjorn3@users.noreply.github.com>
Sat, 27 Jul 2019 15:52:57 +0000 (17:52 +0200)
committerbjorn3 <bjorn3@users.noreply.github.com>
Mon, 29 Jul 2019 09:03:55 +0000 (11:03 +0200)
src/intrinsics.rs

index 8efeb0b4acc391bae54e74c9ae2f7909b06d2a51..3fd1d5fd6d45ee8ccb99910f6edea8e22ecedc0e 100644 (file)
@@ -156,6 +156,25 @@ fn simd_for_each_lane<'tcx, B: Backend>(
 }
 
 macro_rules! simd_binop {
+    ($fx:expr, $intrinsic:expr, icmp($cc:ident, $x:ident, $y:ident) -> $ret:ident) => {
+        simd_for_each_lane($fx, $intrinsic, $x, $y, $ret, |fx, _lane_layout, ret_lane_layout, x_lane, y_lane| {
+            let res_lane = fx.bcx.ins().icmp(IntCC::$cc, x_lane, y_lane);
+            let res_lane = fx.bcx.ins().bint(types::I8, res_lane);
+            CValue::by_val(res_lane, ret_lane_layout)
+        });
+    };
+    ($fx:expr, $intrinsic:expr, icmp($cc_u:ident|$cc_s:ident, $x:ident, $y:ident) -> $ret:ident) => {
+        simd_for_each_lane($fx, $intrinsic, $x, $y, $ret, |fx, lane_layout, ret_lane_layout, x_lane, y_lane| {
+            let res_lane = match lane_layout.ty.sty {
+                ty::Uint(_) => fx.bcx.ins().icmp(IntCC::$cc_u, x_lane, y_lane),
+                ty::Int(_) => fx.bcx.ins().icmp(IntCC::$cc_s, x_lane, y_lane),
+                _ => unreachable!("{:?}", lane_layout.ty),
+            };
+            let res_lane = fx.bcx.ins().bint(types::I8, res_lane);
+            CValue::by_val(res_lane, ret_lane_layout)
+        });
+    };
+
     ($fx:expr, $intrinsic:expr, $op:ident($x:ident, $y:ident) -> $ret:ident) => {
         simd_for_each_lane($fx, $intrinsic, $x, $y, $ret, |fx, _lane_layout, ret_lane_layout, x_lane, y_lane| {
             let res_lane = fx.bcx.ins().$op(x_lane, y_lane);
@@ -753,6 +772,25 @@ fn swap(bcx: &mut FunctionBuilder, v: Value) -> Value {
             ret.write_cvalue(fx, x.unchecked_cast_to(ret.layout()));
         };
 
+        simd_eq, (c x, c y) {
+            simd_binop!(fx, intrinsic, icmp(Equal, x, y) -> ret);
+        };
+        simd_ne, (c x, c y) {
+            simd_binop!(fx, intrinsic, icmp(NotEqual, x, y) -> ret);
+        };
+        simd_lt, (c x, c y) {
+            simd_binop!(fx, intrinsic, icmp(UnsignedLessThan|SignedLessThan, x, y) -> ret);
+        };
+        simd_le, (c x, c y) {
+            simd_binop!(fx, intrinsic, icmp(UnsignedLessThanOrEqual|SignedLessThanOrEqual, x, y) -> ret);
+        };
+        simd_gt, (c x, c y) {
+            simd_binop!(fx, intrinsic, icmp(UnsignedGreaterThan|SignedGreaterThan, x, y) -> ret);
+        };
+        simd_ge, (c x, c y) {
+            simd_binop!(fx, intrinsic, icmp(UnsignedGreaterThanOrEqual|SignedGreaterThanOrEqual, x, y) -> ret);
+        };
+
         simd_add, (c x, c y) {
             simd_binop!(fx, intrinsic, iadd(x, y) -> ret);
         };