| // RUN: %check_clang_tidy %s android-comparison-in-temp-failure-retry %t |
| |
| #define TEMP_FAILURE_RETRY(x) \ |
| ({ \ |
| typeof(x) __z; \ |
| do \ |
| __z = (x); \ |
| while (__z == -1); \ |
| __z; \ |
| }) |
| |
| int foo(); |
| int bar(int a); |
| |
| void test() { |
| int i; |
| TEMP_FAILURE_RETRY((i = foo())); |
| TEMP_FAILURE_RETRY(foo()); |
| TEMP_FAILURE_RETRY((foo())); |
| |
| TEMP_FAILURE_RETRY(foo() == 1); |
| // CHECK-MESSAGES: :[[@LINE-1]]:28: warning: top-level comparison in TEMP_FAILURE_RETRY [android-comparison-in-temp-failure-retry] |
| TEMP_FAILURE_RETRY((foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:29: warning: top-level comparison in TEMP_FAILURE_RETRY |
| TEMP_FAILURE_RETRY((int)(foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:34: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| TEMP_FAILURE_RETRY(bar(foo() == 1)); |
| TEMP_FAILURE_RETRY((bar(foo() == 1))); |
| TEMP_FAILURE_RETRY((bar(foo() == 1)) == 1); |
| // CHECK-MESSAGES: :[[@LINE-1]]:40: warning: top-level comparison in TEMP_FAILURE_RETRY |
| TEMP_FAILURE_RETRY(((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:41: warning: top-level comparison in TEMP_FAILURE_RETRY |
| TEMP_FAILURE_RETRY((int)((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:46: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| #define INDIRECT TEMP_FAILURE_RETRY |
| INDIRECT(foo()); |
| INDIRECT((foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:19: warning: top-level comparison in TEMP_FAILURE_RETRY |
| INDIRECT(bar(foo() == 1)); |
| INDIRECT((int)((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:36: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| #define TFR(x) TEMP_FAILURE_RETRY(x) |
| TFR(foo()); |
| TFR((foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:14: warning: top-level comparison in TEMP_FAILURE_RETRY |
| TFR(bar(foo() == 1)); |
| TFR((int)((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:31: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| #define ADD_TFR(x) (1 + TEMP_FAILURE_RETRY(x) + 1) |
| ADD_TFR(foo()); |
| ADD_TFR(foo() == 1); |
| // CHECK-MESSAGES: :[[@LINE-1]]:17: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| ADD_TFR(bar(foo() == 1)); |
| ADD_TFR((int)((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:35: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| #define ADDP_TFR(x) (1 + TEMP_FAILURE_RETRY((x)) + 1) |
| ADDP_TFR(foo()); |
| ADDP_TFR((foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:19: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| ADDP_TFR(bar(foo() == 1)); |
| ADDP_TFR((int)((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:36: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| #define MACRO TEMP_FAILURE_RETRY(foo() == 1) |
| MACRO; |
| // CHECK-MESSAGES: :[[@LINE-1]]:3: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| // Be sure that being a macro arg doesn't mess with this. |
| #define ID(x) (x) |
| ID(ADDP_TFR(bar(foo() == 1))); |
| ID(ADDP_TFR(bar(foo() == 1) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:31: warning: top-level comparison in TEMP_FAILURE_RETRY |
| ID(MACRO); |
| // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: top-level comparison in TEMP_FAILURE_RETRY |
| |
| #define CMP(x) x == 1 |
| TEMP_FAILURE_RETRY(CMP(foo())); |
| // CHECK-MESSAGES: :[[@LINE-1]]:22: warning: top-level comparison in TEMP_FAILURE_RETRY |
| } |
| |
| // Be sure that it works inside of things like loops, if statements, etc. |
| void control_flow() { |
| do { |
| if (TEMP_FAILURE_RETRY(foo())) { |
| } |
| |
| if (TEMP_FAILURE_RETRY(foo() == 1)) { |
| // CHECK-MESSAGES: :[[@LINE-1]]:34: warning: top-level comparison in TEMP_FAILURE_RETRY |
| } |
| |
| if (TEMP_FAILURE_RETRY(bar(foo() == 1))) { |
| } |
| |
| if (TEMP_FAILURE_RETRY(bar(foo() == 1) == 1)) { |
| // CHECK-MESSAGES: :[[@LINE-1]]:44: warning: top-level comparison in TEMP_FAILURE_RETRY |
| } |
| } while (TEMP_FAILURE_RETRY(foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:37: warning: top-level comparison in TEMP_FAILURE_RETRY |
| } |
| |
| void with_nondependent_variable_type() { |
| #undef TEMP_FAILURE_RETRY |
| #define TEMP_FAILURE_RETRY(x) \ |
| ({ \ |
| long int __z; \ |
| do \ |
| __z = (x); \ |
| while (__z == -1); \ |
| __z; \ |
| }) |
| |
| TEMP_FAILURE_RETRY((foo())); |
| TEMP_FAILURE_RETRY((int)(foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:34: warning: top-level comparison in TEMP_FAILURE_RETRY |
| TEMP_FAILURE_RETRY((bar(foo() == 1))); |
| TEMP_FAILURE_RETRY((int)((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:46: warning: top-level comparison in TEMP_FAILURE_RETRY |
| } |
| |
| // I can't find a case where TEMP_FAILURE_RETRY is implemented like this, but if |
| // we can cheaply support it, I don't see why not. |
| void obscured_temp_failure_retry() { |
| #undef TEMP_FAILURE_RETRY |
| #define IMPL(x) \ |
| ({ \ |
| typeof(x) __z; \ |
| do \ |
| __z = (x); \ |
| while (__z == -1); \ |
| __z; \ |
| }) |
| |
| #define IMPL2(x) IMPL(x) |
| #define TEMP_FAILURE_RETRY(x) IMPL2(x) |
| TEMP_FAILURE_RETRY((foo())); |
| TEMP_FAILURE_RETRY((int)(foo() == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:34: warning: top-level comparison in TEMP_FAILURE_RETRY |
| TEMP_FAILURE_RETRY((bar(foo() == 1))); |
| TEMP_FAILURE_RETRY((int)((bar(foo() == 1)) == 1)); |
| // CHECK-MESSAGES: :[[@LINE-1]]:46: warning: top-level comparison in TEMP_FAILURE_RETRY |
| } |