aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--gcc/c/c-fold.cc3
-rw-r--r--gcc/c/c-parser.cc34
-rw-r--r--gcc/testsuite/gcc.dg/builtin-stdc-rotate-4.c120
3 files changed, 154 insertions, 3 deletions
diff --git a/gcc/c/c-fold.cc b/gcc/c/c-fold.cc
index deb6896..0c4c590 100644
--- a/gcc/c/c-fold.cc
+++ b/gcc/c/c-fold.cc
@@ -410,7 +410,8 @@ c_fully_fold_internal (tree expr, bool in_init, bool *maybe_const_operands,
|| TREE_CODE (TREE_TYPE (orig_op0)) == FIXED_POINT_TYPE)
&& compare_tree_int (op1,
TYPE_PRECISION (TREE_TYPE (orig_op0)))
- >= 0)
+ >= 0
+ && !warning_suppressed_p (expr, OPT_Wshift_count_overflow))
warning_at (loc, OPT_Wshift_count_overflow,
(code == LSHIFT_EXPR
? G_("left shift count >= width of type")
diff --git a/gcc/c/c-parser.cc b/gcc/c/c-parser.cc
index 70fbf94..def6b30 100644
--- a/gcc/c/c-parser.cc
+++ b/gcc/c/c-parser.cc
@@ -12638,8 +12638,38 @@ c_parser_postfix_expression (c_parser *parser)
build_int_cst (utype, prec));
}
- expr.value = build2_loc (loc, code, TREE_TYPE (arg1), arg1,
- arg2);
+ /* The middle-end isn't prepared to handle {L,R}ROTATE_EXPR
+ on types without mode precision, except for large/huge
+ _BitInt types. */
+ if (type_has_mode_precision_p (TREE_TYPE (arg1))
+ || (TREE_CODE (TREE_TYPE (arg1)) == BITINT_TYPE
+ && prec > MAX_FIXED_MODE_SIZE))
+ expr.value = build2_loc (loc, code, TREE_TYPE (arg1), arg1,
+ arg2);
+ else
+ {
+ arg2 = save_expr (arg2);
+ tree t1 = build2_loc (loc, (code == LROTATE_EXPR
+ ? LSHIFT_EXPR : RSHIFT_EXPR),
+ TREE_TYPE (arg1), arg1, arg2);
+ tree t2 = build2_loc (loc, MINUS_EXPR,
+ TREE_TYPE (arg2),
+ build_int_cst (TREE_TYPE (arg2),
+ prec), arg2);
+ t2 = build2_loc (loc, (code == LROTATE_EXPR
+ ? RSHIFT_EXPR : LSHIFT_EXPR),
+ TREE_TYPE (arg1), arg1, t2);
+ suppress_warning (t2, OPT_Wshift_count_overflow);
+ tree t3 = build2_loc (loc, BIT_IOR_EXPR,
+ TREE_TYPE (arg1), t1, t2);
+ tree t4 = build2_loc (loc, NE_EXPR, boolean_type_node,
+ arg2,
+ build_zero_cst (TREE_TYPE (arg2)));
+ t4 = build2_loc (loc, COMPOUND_EXPR, boolean_type_node,
+ arg1, t4);
+ expr.value = build3_loc (loc, COND_EXPR,
+ TREE_TYPE (arg1), t4, t3, arg1);
+ }
if (instrument_expr)
expr.value = build2_loc (loc, COMPOUND_EXPR,
TREE_TYPE (expr.value),
diff --git a/gcc/testsuite/gcc.dg/builtin-stdc-rotate-4.c b/gcc/testsuite/gcc.dg/builtin-stdc-rotate-4.c
new file mode 100644
index 0000000..590b3ab
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/builtin-stdc-rotate-4.c
@@ -0,0 +1,120 @@
+/* PR c/117456 */
+/* { dg-do run } */
+/* { dg-options "-std=c11" } */
+
+struct S {
+ unsigned s : 5;
+};
+
+unsigned
+f1 (struct S s)
+{
+ return __builtin_stdc_rotate_left (s.s, 3);
+}
+
+unsigned
+f2 (struct S s, int n)
+{
+ return __builtin_stdc_rotate_left (s.s, n);
+}
+
+unsigned
+f3 (struct S s)
+{
+ return __builtin_stdc_rotate_right (s.s, 2);
+}
+
+unsigned
+f4 (struct S s, int n)
+{
+ return __builtin_stdc_rotate_right (s.s, n);
+}
+
+#if __BITINT_MAXWIDTH__ >= 64
+unsigned _BitInt(5)
+f5 (unsigned _BitInt(5) s)
+{
+ return __builtin_stdc_rotate_left (s, 3);
+}
+
+unsigned _BitInt(5)
+f6 (unsigned _BitInt(5) s, int n)
+{
+ return __builtin_stdc_rotate_left (s, n);
+}
+
+unsigned _BitInt(5)
+f7 (unsigned _BitInt(5) s)
+{
+ return __builtin_stdc_rotate_right (s, 2);
+}
+
+unsigned _BitInt(5)
+f8 (unsigned _BitInt(5) s, int n)
+{
+ return __builtin_stdc_rotate_right (s, n);
+}
+#endif
+
+#if __BITINT_MAXWIDTH__ >= 125
+unsigned _BitInt(125)
+f9 (unsigned _BitInt(125) s)
+{
+ return __builtin_stdc_rotate_left (s, 13);
+}
+
+unsigned _BitInt(125)
+f10 (unsigned _BitInt(125) s, int n)
+{
+ return __builtin_stdc_rotate_left (s, n);
+}
+
+unsigned _BitInt(125)
+f11 (unsigned _BitInt(125) s)
+{
+ return __builtin_stdc_rotate_right (s, 42);
+}
+
+unsigned _BitInt(125)
+f12 (unsigned _BitInt(125) s, int n)
+{
+ return __builtin_stdc_rotate_right (s, n);
+}
+#endif
+
+int
+main ()
+{
+ struct S s = { 0x12 };
+ if (f1 (s) != 0x14
+ || f2 (s, 0) != 0x12
+ || f2 (s, 2) != 0xa
+ || f2 (s, 1) != 0x5
+ || f3 (s) != 0x14
+ || f4 (s, 0) != 0x12
+ || f4 (s, 2) != 0x14
+ || f4 (s, 1) != 0x9)
+ __builtin_abort ();
+#if __BITINT_MAXWIDTH__ >= 64
+ if (f5 (0x12uwb) != 0x14uwb
+ || f6 (0x12uwb, 0) != 0x12uwb
+ || f6 (0x12uwb, 2) != 0xauwb
+ || f6 (0x12uwb, 1) != 0x5uwb
+ || f7 (0x12uwb) != 0x14uwb
+ || f8 (0x12uwb, 0) != 0x12uwb
+ || f8 (0x12uwb, 2) != 0x14uwb
+ || f8 (0x12uwb, 1) != 0x9uwb)
+ __builtin_abort ();
+#endif
+#if __BITINT_MAXWIDTH__ >= 125
+ if (f9 (12107255122146692213464668179507246062uwb) != 32859299037257821061785486091897129243uwb
+ || f10 (12107255122146692213464668179507246062uwb, 0) != 12107255122146692213464668179507246062uwb
+ || f10 (12107255122146692213464668179507246062uwb, 57) != 786310737972746809290227161460052307uwb
+ || f10 (12107255122146692213464668179507246062uwb, 1) != 24214510244293384426929336359014492124uwb
+ || f11 (12107255122146692213464668179507246062uwb) != 25567301336572975565218391744704605699uwb
+ || f12 (12107255122146692213464668179507246062uwb, 0) != 12107255122146692213464668179507246062uwb
+ || f12 (12107255122146692213464668179507246062uwb, 22) != 27217840477347696606051931660144451082uwb
+ || f12 (12107255122146692213464668179507246062uwb, 1) != 6053627561073346106732334089753623031uwb)
+ __builtin_abort ();
+#endif
+}