bits_type,
in_range, res, bits);
}
+
+/* Test the bit NUMBIT in BITS.
+
+ NUMBIT is one based and counts bits from least significative to most
+ significative, i.e. from "right" to "left". If NUMBIT is not in range then
+ the operator yields false. */
+
+tree
+a68_bits_test (MOID_T *m ATTRIBUTE_UNUSED,
+ tree bits, tree numbit, location_t loc)
+{
+ tree bits_type = TREE_TYPE (bits);
+ tree int_type = TREE_TYPE (numbit);
+
+ bits = save_expr (bits);
+ numbit = save_expr (numbit);
+
+ tree numbit_minus_one = fold_build2 (MINUS_EXPR, int_type,
+ numbit, build_one_cst (int_type));
+ tree mask = fold_build2 (BIT_AND_EXPR, bits_type,
+ bits,
+ fold_build2 (LSHIFT_EXPR,
+ bits_type,
+ build_one_cst (bits_type),
+ fold_convert (bits_type, numbit_minus_one)));
+ tree res = fold_build2 (NE_EXPR,
+ a68_bool_type,
+ fold_build2 (BIT_AND_EXPR, bits_type, bits, mask),
+ build_int_cst (bits_type, 0));
+ tree in_range = fold_build2 (TRUTH_AND_EXPR,
+ int_type,
+ fold_build2 (GE_EXPR, int_type,
+ numbit, build_int_cst (int_type, 1)),
+ fold_build2 (LE_EXPR, int_type,
+ numbit, a68_bits_width (bits_type)));
+
+ return fold_build3_loc (loc, COND_EXPR,
+ a68_bool_type,
+ in_range, res, build_zero_cst (a68_bool_type));
+}
return a68_bits_clear (MOID (p), op1, op2, a68_get_node_location (p));
}
+tree
+a68_lower_test3 (NODE_T *p, LOW_CTX_T ctx)
+{
+ tree op1 = a68_lower_tree (SUB (p), ctx);
+ tree op2 = a68_lower_tree (NEXT (NEXT (SUB (p))), ctx);
+ return a68_bits_test (MOID (p), op1, op2, a68_get_node_location (p));
+}
+
tree
a68_lower_pow_int (NODE_T *p, LOW_CTX_T ctx)
{
a68_prio ("ELEMS", 8);
a68_prio ("SET", 7);
a68_prio ("CLEAR", 7);
+ a68_prio ("TEST", 7);
/* Identifiers. */
a68_idf (A68_EXT, "infinity", M_REAL, a68_lower_infinity);
a68_idf (A68_EXT, "minusinfinity", M_REAL, a68_lower_minusinfinity);
m = a68_proc (M_SHORT_SHORT_BITS, M_SHORT_SHORT_BITS, M_INT, NO_MOID);
a68_op (A68_EXT, "SET", m, a68_lower_set3);
a68_op (A68_EXT, "CLEAR", m, a68_lower_clear3);
+ m = a68_proc (M_BOOL, M_SHORT_SHORT_BITS, M_INT, NO_MOID);
+ a68_op (A68_EXT, "TEST", m, a68_lower_test3);
/* SHORT BITS operators. */
m = a68_proc (M_SHORT_BITS, M_SHORT_BITS, M_INT, NO_MOID);
a68_op (A68_EXT, "SET", m, a68_lower_set3);
a68_op (A68_EXT, "CLEAR", m, a68_lower_clear3);
+ m = a68_proc (M_BOOL, M_SHORT_BITS, M_INT, NO_MOID);
+ a68_op (A68_EXT, "TEST", m, a68_lower_test3);
/* BITS operators. */
m = a68_proc (M_BITS, M_BITS, M_INT, NO_MOID);
a68_op (A68_EXT, "SET", m, a68_lower_set3);
a68_op (A68_EXT, "CLEAR", m, a68_lower_clear3);
+ m = a68_proc (M_BOOL, M_BITS, M_INT, NO_MOID);
+ a68_op (A68_EXT, "TEST", m, a68_lower_test3);
/* LONG BITS operators. */
m = a68_proc (M_LONG_BITS, M_LONG_BITS, M_INT, NO_MOID);
a68_op (A68_EXT, "SET", m, a68_lower_set3);
a68_op (A68_EXT, "CLEAR", m, a68_lower_clear3);
+ m = a68_proc (M_BOOL, M_LONG_BITS, M_INT, NO_MOID);
+ a68_op (A68_EXT, "TEST", m, a68_lower_test3);
/* LONG LONG BITS operators. */
m = a68_proc (M_LONG_LONG_BITS, M_LONG_LONG_BITS, M_INT, NO_MOID);
a68_op (A68_EXT, "SET", m, a68_lower_set3);
a68_op (A68_EXT, "CLEAR", m, a68_lower_clear3);
+ m = a68_proc (M_BOOL, M_LONG_LONG_BITS, M_INT, NO_MOID);
+ a68_op (A68_EXT, "TEST", m, a68_lower_test3);
}
/* POSIX prelude. */
tree a68_bits_ne (tree a, tree b, location_t loc = UNKNOWN_LOCATION);
tree a68_bits_set (MOID_T *m, tree bits, tree numbit, location_t loc = UNKNOWN_LOCATION);
tree a68_bits_clear (MOID_T *m, tree bits, tree numbit, location_t loc = UNKNOWN_LOCATION);
+tree a68_bits_test (MOID_T *m, tree bits, tree numbit, location_t loc = UNKNOWN_LOCATION);
/* a68-low_bools.cc */
tree a68_lower_longlongrandom (NODE_T *p, LOW_CTX_T ctx);
tree a68_lower_set3 (NODE_T *p, LOW_CTX_T ctx);
tree a68_lower_clear3 (NODE_T *p, LOW_CTX_T ctx);
+tree a68_lower_test3 (NODE_T *p, LOW_CTX_T ctx);
tree a68_lower_posixargc (NODE_T *p, LOW_CTX_T ctx);
tree a68_lower_posixargv (NODE_T *p, LOW_CTX_T ctx);
tree a68_lower_posixputchar (NODE_T *p, LOW_CTX_T ctx);
then the operator yields @B{b}.
@end deftypefn
+@deftypefn Operator {} {@B{test}} {= (@B{l} @B{bits} b, @B{int} n) @B{bool}}
+Dyadic operator that tests whether the @code{n}th least significant
+bit in @code{@B{b}} is set. If @code{n} is not in the range
+@code{1,L_bits_width]} then the operator yields @B{false}.
+@end deftypefn
+
@node Extended math procedures
@section Extended math procedures
--- /dev/null
+begin assert (NOT (16rff TEST 9));
+ assert (NOT (16rff TEST 0));
+ assert (NOT (16rff TEST -1));
+ assert (2r100 TEST 3)
+end