diff options
Diffstat (limited to 'clang/test')
| -rw-r--r-- | clang/test/CodeGen/no-builtin.cpp | 64 | ||||
| -rw-r--r-- | clang/test/Misc/pragma-attribute-supported-attributes-list.test | 1 | ||||
| -rw-r--r-- | clang/test/Sema/no-builtin.cpp | 51 |
3 files changed, 116 insertions, 0 deletions
diff --git a/clang/test/CodeGen/no-builtin.cpp b/clang/test/CodeGen/no-builtin.cpp new file mode 100644 index 00000000000..3c5d681282d --- /dev/null +++ b/clang/test/CodeGen/no-builtin.cpp @@ -0,0 +1,64 @@ +// RUN: %clang_cc1 -triple x86_64-linux-gnu -S -emit-llvm -o - %s | FileCheck %s + +// CHECK-LABEL: define void @foo_no_mempcy() #0 +extern "C" void foo_no_mempcy() __attribute__((no_builtin("memcpy"))) {} + +// CHECK-LABEL: define void @foo_no_mempcy_twice() #0 +extern "C" void foo_no_mempcy_twice() __attribute__((no_builtin("memcpy"))) __attribute__((no_builtin("memcpy"))) {} + +// CHECK-LABEL: define void @foo_no_builtins() #1 +extern "C" void foo_no_builtins() __attribute__((no_builtin)) {} + +// CHECK-LABEL: define void @foo_no_mempcy_memset() #2 +extern "C" void foo_no_mempcy_memset() __attribute__((no_builtin("memset", "memcpy"))) {} + +// CHECK-LABEL: define void @separate_attrs() #2 +extern "C" void separate_attrs() __attribute__((no_builtin("memset"))) __attribute__((no_builtin("memcpy"))) {} + +// CHECK-LABEL: define void @separate_attrs_ordering() #2 +extern "C" void separate_attrs_ordering() __attribute__((no_builtin("memcpy"))) __attribute__((no_builtin("memset"))) {} + +struct A { + virtual int foo() const __attribute__((no_builtin("memcpy"))) { return 1; } + virtual ~A(); +}; + +struct B : public A { + int foo() const override __attribute__((no_builtin("memmove"))) { return 2; } + virtual ~B(); +}; + +// CHECK-LABEL: define void @call_a_foo(%struct.A* %a) #3 +extern "C" void call_a_foo(A *a) { + // CHECK: %call = call i32 %2(%struct.A* %0) + a->foo(); // virtual call is not annotated +} + +// CHECK-LABEL: define void @call_b_foo(%struct.B* %b) #3 +extern "C" void call_b_foo(B *b) { + // CHECK: %call = call i32 %2(%struct.B* %0) + b->foo(); // virtual call is not annotated +} + +// CHECK-LABEL: define void @call_foo_no_mempcy() #3 +extern "C" void call_foo_no_mempcy() { + // CHECK: call void @foo_no_mempcy() #6 + foo_no_mempcy(); // call gets annotated with "no-builtin-memcpy" +} + +A::~A() {} // Anchoring A so A::foo() gets generated +B::~B() {} // Anchoring B so B::foo() gets generated + +// CHECK-LABEL: define linkonce_odr i32 @_ZNK1A3fooEv(%struct.A* %this) unnamed_addr #0 comdat align 2 +// CHECK-LABEL: define linkonce_odr i32 @_ZNK1B3fooEv(%struct.B* %this) unnamed_addr #5 comdat align 2 + +// CHECK: attributes #0 = {{{.*}}"no-builtin-memcpy"{{.*}}} +// CHECK-NOT: attributes #0 = {{{.*}}"no-builtin-memmove"{{.*}}} +// CHECK-NOT: attributes #0 = {{{.*}}"no-builtin-memset"{{.*}}} +// CHECK: attributes #1 = {{{.*}}"no-builtins"{{.*}}} +// CHECK: attributes #2 = {{{.*}}"no-builtin-memcpy"{{.*}}"no-builtin-memset"{{.*}}} +// CHECK-NOT: attributes #2 = {{{.*}}"no-builtin-memmove"{{.*}}} +// CHECK: attributes #5 = {{{.*}}"no-builtin-memmove"{{.*}}} +// CHECK-NOT: attributes #5 = {{{.*}}"no-builtin-memcpy"{{.*}}} +// CHECK-NOT: attributes #5 = {{{.*}}"no-builtin-memset"{{.*}}} +// CHECK: attributes #6 = { "no-builtin-memcpy" } diff --git a/clang/test/Misc/pragma-attribute-supported-attributes-list.test b/clang/test/Misc/pragma-attribute-supported-attributes-list.test index 25802bd73c5..729e9b7a6f7 100644 --- a/clang/test/Misc/pragma-attribute-supported-attributes-list.test +++ b/clang/test/Misc/pragma-attribute-supported-attributes-list.test @@ -75,6 +75,7 @@ // CHECK-NEXT: NSConsumed (SubjectMatchRule_variable_is_parameter) // CHECK-NEXT: NSConsumesSelf (SubjectMatchRule_objc_method) // CHECK-NEXT: Naked (SubjectMatchRule_function) +// CHECK-NEXT: NoBuiltin (SubjectMatchRule_function) // CHECK-NEXT: NoCommon (SubjectMatchRule_variable) // CHECK-NEXT: NoDebug (SubjectMatchRule_type_alias, SubjectMatchRule_hasType_functionType, SubjectMatchRule_objc_method, SubjectMatchRule_variable_not_is_parameter) // CHECK-NEXT: NoDestroy (SubjectMatchRule_variable) diff --git a/clang/test/Sema/no-builtin.cpp b/clang/test/Sema/no-builtin.cpp new file mode 100644 index 00000000000..40781abd303 --- /dev/null +++ b/clang/test/Sema/no-builtin.cpp @@ -0,0 +1,51 @@ +// RUN: %clang_cc1 -triple x86_64-unknown-unknown -fsyntax-only -verify %s + +/// Prevent use of all builtins. +void valid_attribute_all_1() __attribute__((no_builtin)) {} +void valid_attribute_all_2() __attribute__((no_builtin())) {} + +/// Prevent use of specific builtins. +void valid_attribute_function() __attribute__((no_builtin("memcpy"))) {} +void valid_attribute_functions() __attribute__((no_builtin("memcpy"))) __attribute__((no_builtin("memcmp"))) {} + +/// Many times the same builtin is fine. +void many_attribute_function_1() __attribute__((no_builtin)) __attribute__((no_builtin)) {} +void many_attribute_function_2() __attribute__((no_builtin("memcpy"))) __attribute__((no_builtin("memcpy"))) {} +void many_attribute_function_3() __attribute__((no_builtin("memcpy", "memcpy"))) {} +void many_attribute_function_4() __attribute__((no_builtin("memcpy", "memcpy"))) __attribute__((no_builtin("memcpy"))) {} + +/// Invalid builtin name. +void invalid_builtin() __attribute__((no_builtin("not_a_builtin"))) {} +// expected-warning@-1 {{'not_a_builtin' is not a valid builtin name for no_builtin}} + +/// Can't use bare no_builtin with a named one. +void wildcard_and_functionname() __attribute__((no_builtin)) __attribute__((no_builtin("memcpy"))) {} +// expected-error@-1 {{empty no_builtin cannot be composed with named ones}} + +/// Can't attach attribute to a variable. +int __attribute__((no_builtin)) variable; +// expected-warning@-1 {{'no_builtin' attribute only applies to functions}} + +/// Can't attach attribute to a declaration. +void nobuiltin_on_declaration() __attribute__((no_builtin)); +// expected-error@-1 {{no_builtin attribute is permitted on definitions only}} + +struct S { + /// Can't attach attribute to a defaulted function, + S() + __attribute__((no_builtin)) = default; + // expected-error@-1 {{no_builtin attribute has no effect on defaulted or deleted functions}} + + /// Can't attach attribute to a deleted function, + S(const S &) + __attribute__((no_builtin)) = delete; + // expected-error@-1 {{no_builtin attribute has no effect on defaulted or deleted functions}} + + void whatever() __attribute__((no_builtin("memcpy"))); + // expected-error@-1 {{no_builtin attribute is permitted on definitions only}} +}; + +/// Can't attach attribute to an aliased function. +void alised_function() {} +void aliasing_function() __attribute__((no_builtin)) __attribute__((alias("alised_function"))); +// expected-error@-1 {{no_builtin attribute is permitted on definitions only}} |

