1 | // RUN: %clang_cc1 -fsyntax-only -Wextra-semi-stmt -verify %s |
2 | // RUN: cp %s %t |
3 | // RUN: %clang_cc1 -x c++ -Wextra-semi-stmt -fixit %t |
4 | // RUN: %clang_cc1 -x c++ -Wextra-semi-stmt -Werror %t |
5 | |
6 | #define GOODMACRO(varname) int varname |
7 | #define BETTERMACRO(varname) GOODMACRO(varname); |
8 | #define NULLMACRO(varname) |
9 | |
10 | enum MyEnum { |
11 | E1, |
12 | E2 |
13 | }; |
14 | |
15 | void test() { |
16 | ; // expected-warning {{empty expression statement has no effect; remove unnecessary ';' to silence this warning}} |
17 | ; |
18 | |
19 | // This removal of extra semi also consumes all the comments. |
20 | // clang-format: off |
21 | ;;; |
22 | // clang-format: on |
23 | |
24 | // clang-format: off |
25 | ;NULLMACRO(ZZ); |
26 | // clang-format: on |
27 | |
28 | {}; // expected-warning {{empty expression statement has no effect; remove unnecessary ';' to silence this warning}} |
29 | |
30 | { |
31 | ; // expected-warning {{empty expression statement has no effect; remove unnecessary ';' to silence this warning}} |
32 | } |
33 | |
34 | if (true) { |
35 | ; // expected-warning {{empty expression statement has no effect; remove unnecessary ';' to silence this warning}} |
36 | } |
37 | |
38 | GOODMACRO(v0); // OK |
39 | |
40 | GOODMACRO(v1;) // OK |
41 | |
42 | BETTERMACRO(v2) // OK |
43 | |
44 | BETTERMACRO(v3;) // Extra ';', but within macro expansion, so ignored. |
45 | |
46 | BETTERMACRO(v4); // expected-warning {{empty expression statement has no effect; remove unnecessary ';' to silence this warning}} |
47 | |
48 | BETTERMACRO(v5;); // expected-warning {{empty expression statement has no effect; remove unnecessary ';' to silence this warning}} |
49 | |
50 | NULLMACRO(v6) // OK |
51 | |
52 | NULLMACRO(v7); // OK. This could be either GOODMACRO() or BETTERMACRO() situation, so we can't know we can drop it. |
53 | |
54 | if (true) |
55 | ; // OK |
56 | |
57 | while (true) |
58 | ; // OK |
59 | |
60 | do |
61 | ; // OK |
62 | while (true); |
63 | |
64 | for (;;) // OK |
65 | ; // OK |
66 | |
67 | MyEnum my_enum; |
68 | switch (my_enum) { |
69 | case E1: |
70 | // stuff |
71 | break; |
72 | case E2:; // OK |
73 | } |
74 | |
75 | for (;;) { |
76 | for (;;) { |
77 | goto contin_outer; |
78 | } |
79 | contin_outer:; // OK |
80 | } |
81 | } |
82 | |
83 | ; |
84 | |
85 | namespace NS {}; |
86 | |
87 | void foo(int x) { |
88 | switch (x) { |
89 | case 0: |
90 | [[fallthrough]]; |
91 | case 1: |
92 | return; |
93 | } |
94 | } |
95 | |
96 | [[]]; |
97 | |