diff options
| author | Richard Smith <richard-llvm@metafoo.co.uk> | 2016-03-08 22:17:41 +0000 |
|---|---|---|
| committer | Richard Smith <richard-llvm@metafoo.co.uk> | 2016-03-08 22:17:41 +0000 |
| commit | 872307e2ac6f9a1e8a83e23a5dc0db88538fa860 (patch) | |
| tree | ab727fb967e4d216dc5456c84842a0850eb21f1a /clang/test/CodeGenCXX/cxx1z-initializer-aggregate.cpp | |
| parent | a99000dd311b972e35f889c61bbdbc22a1680abd (diff) | |
| download | bcm5719-llvm-872307e2ac6f9a1e8a83e23a5dc0db88538fa860.tar.gz bcm5719-llvm-872307e2ac6f9a1e8a83e23a5dc0db88538fa860.zip | |
P0017R1: In C++1z, an aggregate class can have (public non-virtual) base classes; these are initialized as if they were data members.
llvm-svn: 262963
Diffstat (limited to 'clang/test/CodeGenCXX/cxx1z-initializer-aggregate.cpp')
| -rw-r--r-- | clang/test/CodeGenCXX/cxx1z-initializer-aggregate.cpp | 114 |
1 files changed, 114 insertions, 0 deletions
diff --git a/clang/test/CodeGenCXX/cxx1z-initializer-aggregate.cpp b/clang/test/CodeGenCXX/cxx1z-initializer-aggregate.cpp new file mode 100644 index 00000000000..9110e49f93a --- /dev/null +++ b/clang/test/CodeGenCXX/cxx1z-initializer-aggregate.cpp @@ -0,0 +1,114 @@ +// RUN: %clang_cc1 -std=c++1z %s -triple x86_64-linux-gnu -fexceptions -fcxx-exceptions -emit-llvm -o - | FileCheck %s + +namespace Constant { + struct A { + int n; + char k; + ~A(); + }; + + struct B { + char k2; + }; + + struct C : B {}; + + struct D : A, C {}; + + C c1 = {}; + C c2 = {1}; + // CHECK: @_ZN8Constant2c1E = global { i8 } zeroinitializer, align 1 + // CHECK: @_ZN8Constant2c2E = global { i8 } { i8 1 }, align 1 + + // Test packing bases into tail padding. + D d1 = {}; + D d2 = {1, 2, 3}; + D d3 = {1}; + // CHECK: @_ZN8Constant2d1E = global { i32, i8, i8 } zeroinitializer, align 4 + // CHECK: @_ZN8Constant2d2E = global { i32, i8, i8 } { i32 1, i8 2, i8 3 }, align 4 + // CHECK: @_ZN8Constant2d3E = global { i32, i8, i8 } { i32 1, i8 0, i8 0 }, align 4 + + // CHECK-LABEL: define {{.*}}global_var_init + // CHECK: call {{.*}} @__cxa_atexit({{.*}} @_ZN8Constant1DD1Ev {{.*}} @_ZN8Constant2d1E + + // CHECK-LABEL: define {{.*}}global_var_init + // CHECK: call {{.*}} @__cxa_atexit({{.*}} @_ZN8Constant1DD1Ev {{.*}} @_ZN8Constant2d2E + + // CHECK-LABEL: define {{.*}}global_var_init + // CHECK: call {{.*}} @__cxa_atexit({{.*}} @_ZN8Constant1DD1Ev {{.*}} @_ZN8Constant2d3E +} + +namespace Dynamic { + struct A { + A(); + A(int); + A(const char*, unsigned); + ~A(); + void *p; + }; + + struct B { + ~B(); + int n = 5; + }; + + struct C { + C(bool = true); + }; + + int f(), g(), h(), i(); + struct D : A, B, C { + int n = f(); + }; + + D d1 = {}; + // CHECK-LABEL: define {{.*}}global_var_init + // CHECK: call void @_ZN7Dynamic1AC2Ev({{.*}} @_ZN7Dynamic2d1E + // CHECK: store i32 5, {{.*}}i8* getelementptr inbounds {{.*}} @_ZN7Dynamic2d1E{{.*}}, i64 8 + // CHECK: invoke void @_ZN7Dynamic1CC2Eb({{.*}} @_ZN7Dynamic2d1E{{.*}}, i1 zeroext true) + // CHECK: unwind label %[[UNWIND:.*]] + // CHECK: invoke i32 @_ZN7Dynamic1fEv() + // CHECK: unwind label %[[UNWIND:.*]] + // CHECK: store i32 {{.*}}, i32* getelementptr {{.*}} @_ZN7Dynamic2d1E, i32 0, i32 2 + // CHECK: call {{.*}} @__cxa_atexit({{.*}} @_ZN7Dynamic1DD1Ev {{.*}} @_ZN7Dynamic2d1E + // CHECK: ret + // + // UNWIND: + // CHECK: call void @_ZN7Dynamic1BD1Ev({{.*}}i8* getelementptr inbounds {{.*}} @_ZN7Dynamic2d1E{{.*}}, i64 8 + // CHECK: call void @_ZN7Dynamic1AD1Ev({{.*}} @_ZN7Dynamic2d1E + + D d2 = {1, 2, false}; + // CHECK-LABEL: define {{.*}}global_var_init + // CHECK: call void @_ZN7Dynamic1AC1Ei({{.*}} @_ZN7Dynamic2d2E{{.*}}, i32 1) + // CHECK: store i32 2, {{.*}}i8* getelementptr inbounds {{.*}}@_ZN7Dynamic2d2E{{.*}}, i64 8 + // CHECK: invoke void @_ZN7Dynamic1CC1Eb({{.*}} @_ZN7Dynamic2d2E{{.*}}, i1 zeroext false) + // CHECK: invoke i32 @_ZN7Dynamic1fEv() + // CHECK: store i32 {{.*}}, i32* getelementptr {{.*}} @_ZN7Dynamic2d2E, i32 0, i32 2 + // CHECK: call {{.*}} @__cxa_atexit({{.*}} @_ZN7Dynamic1DD1Ev {{.*}} @_ZN7Dynamic2d2E + // CHECK: ret void + + D d3 = {g(), h(), {}, i()}; + // CHECK-LABEL: define {{.*}}global_var_init + // CHECK: %[[G_CALL:.*]] = call i32 @_ZN7Dynamic1gEv() + // CHECK: call void @_ZN7Dynamic1AC1Ei({{.*}} @_ZN7Dynamic2d3E{{.*}}, i32 %[[G_CALL]]) + // CHECK: %[[H_CALL:.*]] = invoke i32 @_ZN7Dynamic1hEv() + // CHECK: unwind label %[[DESTROY_A_LPAD:.*]] + // CHECK: store i32 %[[H_CALL]], {{.*}}i8* getelementptr inbounds {{.*}} @_ZN7Dynamic2d3E{{.*}}, i64 8 + // CHECK: invoke void @_ZN7Dynamic1CC2Eb({{.*}} @_ZN7Dynamic2d3E{{.*}}, i1 zeroext true) + // CHECK: unwind label %[[DESTROY_AB_LPAD:.*]] + // CHECK: %[[I_CALL:.*]] = invoke i32 @_ZN7Dynamic1iEv() + // CHECK: unwind label %[[DESTROY_AB_LPAD:.*]] + // CHECK: store i32 %[[I_CALL]], i32* getelementptr {{.*}} @_ZN7Dynamic2d3E, i32 0, i32 2 + // CHECK: call {{.*}} @__cxa_atexit({{.*}} @_ZN7Dynamic1DD1Ev {{.*}} @_ZN7Dynamic2d3E to i8* + // CHECK: ret + // + // DESTROY_A_LPAD: + // CHECK: br label %[[A_CLEANUP:.*]] + // + // DESTROY_B_LPAD: + // CHECK: call void @_ZN7Dynamic1BD1Ev({{.*}}i8* getelementptr inbounds {{.*}} @_ZN7Dynamic2d3E{{.*}}, i64 8 + // CHECK: br label %[[A_CLEANUP:.*]] + // + // A_CLEANUP: + // CHECK: call void @_ZN7Dynamic1AD1Ev({{.*}} @_ZN7Dynamic2d3E +} |

