+2019-11-15 Joseph Myers <joseph@codesourcery.com>
+
+ * c-attribs.c (handle_fallthrough_attribute): Remove static.
+ * c-common.h (handle_fallthrough_attribute): Declare.
+
2019-11-15 Joseph Myers <joseph@codesourcery.com>
* c-attribs.c (handle_deprecated_attribute): Remove static.
static tree handle_omp_declare_target_attribute (tree *, tree, tree, int,
bool *);
static tree handle_designated_init_attribute (tree *, tree, tree, int, bool *);
-static tree handle_fallthrough_attribute (tree *, tree, tree, int, bool *);
static tree handle_patchable_function_entry_attribute (tree *, tree, tree,
int, bool *);
static tree handle_copy_attribute (tree *, tree, tree, int, bool *);
/* Handle a "fallthrough" attribute; arguments as in struct
attribute_spec.handler. */
-static tree
+tree
handle_fallthrough_attribute (tree *, tree name, tree, int,
bool *no_add_attrs)
{
extern bool attribute_takes_identifier_p (const_tree);
extern tree handle_deprecated_attribute (tree *, tree, tree, int, bool *);
extern tree handle_unused_attribute (tree *, tree, tree, int, bool *);
+extern tree handle_fallthrough_attribute (tree *, tree, tree, int, bool *);
extern int parse_tm_stmt_attr (tree, int);
extern int tm_attr_to_mask (tree);
extern tree tm_mask_to_attr (int);
+2019-11-15 Joseph Myers <joseph@codesourcery.com>
+
+ * c-decl.c (std_attribute_table): Add fallthrough.
+ * c-parser.c (c_parser_declaration_or_fndef): Diagnose fallthrough
+ attribute at top level.
+
2019-11-15 Joseph Myers <joseph@codesourcery.com>
* c-decl.c (std_attribute_table): New.
affects_type_identity, handler, exclude } */
{ "deprecated", 0, 1, false, false, false, false,
handle_deprecated_attribute, NULL },
+ { "fallthrough", 0, 0, false, false, false, false,
+ handle_fallthrough_attribute, NULL },
{ NULL, 0, 0, false, false, false, false, NULL, NULL }
};
{
if (fallthru_attr_p != NULL)
*fallthru_attr_p = true;
- tree fn = build_call_expr_internal_loc (here, IFN_FALLTHROUGH,
- void_type_node, 0);
- add_stmt (fn);
+ if (nested)
+ {
+ tree fn = build_call_expr_internal_loc (here, IFN_FALLTHROUGH,
+ void_type_node, 0);
+ add_stmt (fn);
+ }
+ else
+ pedwarn (here, OPT_Wattributes,
+ "%<fallthrough%> attribute at top level");
}
else if (empty_ok && !(have_attrs
&& specs->non_std_attrs_seen_p))
+2019-11-15 Joseph Myers <joseph@codesourcery.com>
+
+ * gcc.dg/c2x-attr-fallthrough-2.c,
+ gcc.dg/c2x-attr-fallthrough-3.c: New tests.
+
2019-11-15 Joseph Myers <joseph@codesourcery.com>
* gcc.dg/c2x-attr-deprecated-1.c, gcc.dg/c2x-attr-deprecated-2.c,
--- /dev/null
+/* Test C2x attribute syntax. Invalid use of fallthrough attribute. */
+/* { dg-do compile } */
+/* { dg-options "-std=c2x -pedantic-errors -Wextra" } */
+
+[[fallthrough]]; /* { dg-error "'fallthrough' attribute at top level" } */
+
+int [[fallthrough]] x; /* { dg-warning "ignored" } */
+/* { dg-message "that appertains to a type-specifier" "appertains" { target *-*-* } .-1 } */
+
+int g () [[fallthrough]]; /* { dg-warning "ignored" } */
+/* { dg-message "that appertains to a type-specifier" "appertains" { target *-*-* } .-1 } */
+
+int
+f (int a)
+{
+ [[fallthrough]] int b = 2; /* { dg-warning "not followed by|ignored" } */
+ switch (a)
+ {
+ case 1:
+ b = 1; /* { dg-warning "may fall through" } */
+ case 2:
+ b = 2; /* { dg-warning "may fall through" } */
+ [[fallthrough()]]; /* { dg-error "does not take any arguments" } */
+ case 3:
+ b += 7;
+ break;
+ case 4:
+ b = 4; /* { dg-warning "may fall through" } */
+ [[fallthrough(1)]]; /* { dg-error "does not take any arguments|expected" } */
+ case 5:
+ b += 5;
+ break;
+ }
+ [[fallthrough]] return b; /* { dg-warning "ignored" } */
+}
--- /dev/null
+/* Test C2x attribute syntax. Invalid use of fallthrough attribute
+ outside switch or in bad context inside switch. */
+/* { dg-do compile } */
+/* { dg-options "-std=c2x -pedantic-errors -Wextra" } */
+
+int
+f (int a)
+{
+ [[fallthrough]]; /* { dg-error "invalid use of attribute 'fallthrough'" } */
+ switch (a)
+ {
+ case 1:
+ a++;
+ [[fallthrough]]; /* { dg-warning "attribute 'fallthrough' not preceding a case label or default label" } */
+ a++;
+ }
+ return a;
+}