| // RUN: %clang_cc1 -std=c11 -fsyntax-only -fsanitize=kcfi -verify %s |
| // RUN: %clang_cc1 -std=c89 -DKNR -fsyntax-only -fsanitize=kcfi -verify %s |
| |
| #define __cfi_salt(S) __attribute__((cfi_salt(S))) |
| |
| int bad1(void) __cfi_salt(); // expected-error{{'cfi_salt' attribute takes one argument}} |
| int bad2(void) __cfi_salt(42); // expected-error{{expected string literal as argument of 'cfi_salt' attribute}} |
| int bad3(void) __attribute__((cfi_salt("a", "b", "c"))); // expected-error{{'cfi_salt' attribute takes one argument}} |
| |
| |
| int foo(int a, int b) __cfi_salt("pepper"); // ok |
| int foo(int a, int b) __cfi_salt("pepper"); // ok |
| |
| #ifndef KNR |
| typedef int (*bar_t)(void) __cfi_salt("pepper"); // ok |
| typedef int (*bar_t)(void) __cfi_salt("pepper"); // ok |
| #endif |
| |
| // FIXME: Should we allow this? |
| // int b(void) __cfi_salt("salt 'n") __cfi_salt("pepper"); |
| // bar_t bar_fn __cfi_salt("salt 'n"); |
| |
| int baz __cfi_salt("salt"); // expected-warning{{'cfi_salt' only applies to function types}} |
| |
| int baz_fn(int a, int b) __cfi_salt("salt 'n"); // expected-note{{previous declaration is here}} |
| int baz_fn(int a, int b) __cfi_salt("pepper"); // expected-error{{conflicting types for 'baz_fn'}} |
| |
| int mux_fn(int a, int b) __cfi_salt("salt 'n"); // expected-note{{previous declaration is here}} |
| int mux_fn(int a, int b) __cfi_salt("pepper") { // expected-error{{conflicting types for 'mux_fn'}} |
| return a * b; |
| } |
| |
| typedef int qux_t __cfi_salt("salt"); // expected-warning{{'cfi_salt' only applies to function types}} |
| |
| typedef int (*quux_t)(void) __cfi_salt("salt 'n"); // expected-note{{previous definition is here}} |
| typedef int (*quux_t)(void) __cfi_salt("pepper"); // expected-error{{typedef redefinition with different type}} |
| |
| void func1(int a) __cfi_salt("pepper"); // expected-note{{previous declaration is here}} |
| void func1(int a) { } // expected-error{{conflicting types for 'func1'}} |
| void (*fp1)(int) = func1; // expected-error{{incompatible function pointer types initializing 'void (*)(int)' with an expression of type 'void (int)'}} |
| |
| void func2(int) [[clang::cfi_salt("test")]]; // expected-note{{previous declaration is here}} |
| void func2(int a) { } // expected-error{{conflicting types for 'func2'}} |
| void (*fp2)(int) = func2; // expected-error{{incompatible function pointer types initializing 'void (*)(int)' with an expression of type 'void (int)'}} |
| |
| void func3(int) __cfi_salt("pepper"); // ok |
| void func3(int a) __cfi_salt("pepper") { } // ok |
| void (* __cfi_salt("pepper") fp3)(int) = func3; // ok |
| void (*fp3_noattr)(int) = func3; // expected-error{{incompatible function pointer types initializing 'void (*)(int)' with an expression of type 'void (int)'}} |
| |
| void func4(int) [[clang::cfi_salt("test")]]; // ok |
| void func4(int a) [[clang::cfi_salt("test")]] { } // ok |
| void (* [[clang::cfi_salt("test")]] fp4)(int) = func4; // ok |
| void (*fp4_noattr)(int) = func4; // expected-error{{incompatible function pointer types initializing 'void (*)(int)' with an expression of type 'void (int)'}} |
| |
| #ifdef KNR |
| // K&R C function without a prototype |
| void func() __attribute__((cfi_salt("pepper"))); // expected-error {{attribute only applies to non-K&R-style functions}} |
| void (*fp)() __attribute__((cfi_salt("pepper"))); // expected-error {{attribute only applies to non-K&R-style functions}} |
| #endif |