| // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fclangir -emit-cir -mmlir --mlir-print-ir-before=cir-lowering-prepare %s -o %t.cir 2> %t-before.cir |
| // RUN: FileCheck --check-prefix=CIR-BEFORE-LPP --input-file=%t-before.cir %s |
| // RUN: FileCheck --check-prefix=CIR --input-file=%t.cir %s |
| // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fclangir -emit-cir -O1 %s -o %t-o1.cir |
| // RUN: FileCheck --check-prefix=CIR-O1 --input-file=%t-o1.cir %s |
| // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fclangir -emit-llvm %s -o %t-cir.ll |
| // RUN: FileCheck --check-prefix=LLVM --input-file=%t-cir.ll %s |
| // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -emit-llvm %s -o %t.ll |
| // RUN: FileCheck --check-prefix=OGCG --input-file=%t.ll %s |
| // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fclangir -emit-llvm -O1 -disable-llvm-passes %s -o %t-opt.ll |
| // RUN: FileCheck --check-prefix=LLVM-OPT --input-file=%t-opt.ll %s |
| // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -emit-llvm -O1 -disable-llvm-passes %s -o %t-opt-ogcg.ll |
| // RUN: FileCheck --check-prefix=OGCG-OPT --input-file=%t-opt-ogcg.ll %s |
| |
| // Test for global with constant storage - const object with constructor but no destructor |
| // Check that we add an llvm.invariant.start to mark when a global becomes read-only. |
| |
| struct A { |
| A(); |
| int n; |
| }; |
| |
| // Should emit invariant.start - has constructor, no destructor, no mutable |
| extern const A a = A(); |
| |
| struct A2 { |
| A2(); |
| constexpr ~A2() {} |
| int n; |
| }; |
| |
| // Should emit invariant.start - constexpr destructor doesn't prevent constant storage |
| extern const A2 a2 = A2(); |
| |
| struct B { |
| B(); |
| mutable int n; |
| }; |
| |
| // Should NOT emit invariant.start - has mutable member |
| extern const B b = B(); |
| |
| // Simple case - just const C c; (no initializer) - Andy's suggestion |
| class C { |
| public: |
| C(); |
| int a; |
| int b; |
| }; |
| |
| const C c; |
| |
| // CIR checks before LoweringPrepare transformation - globals have ctor regions |
| // Test case 'a' - before LoweringPrepare |
| // CIR-BEFORE-LPP: cir.global external @a = ctor : !rec_A { |
| // CIR-BEFORE-LPP: %[[OBJ:.*]] = cir.get_global @a : !cir.ptr<!rec_A> |
| // CIR-BEFORE-LPP: cir.call @_ZN1AC1Ev(%[[OBJ]]) : (!cir.ptr<!rec_A> {{.*}}) -> () |
| // CIR-BEFORE-LPP: %[[OBJ2:.*]] = cir.get_global @a : !cir.ptr<!rec_A> |
| // CIR-BEFORE-LPP: } |
| |
| // Test case 'a2' - before LoweringPrepare |
| // CIR-BEFORE-LPP: cir.global external @a2 = ctor : !rec_A2 { |
| // CIR-BEFORE-LPP: %[[OBJ:.*]] = cir.get_global @a2 : !cir.ptr<!rec_A2> |
| // CIR-BEFORE-LPP: cir.call @_ZN2A2C1Ev(%[[OBJ]]) : (!cir.ptr<!rec_A2> {{.*}}) -> () |
| // CIR-BEFORE-LPP: %[[OBJ2:.*]] = cir.get_global @a2 : !cir.ptr<!rec_A2> |
| // CIR-BEFORE-LPP: } |
| |
| // Test case 'b' - before LoweringPrepare |
| // CIR-BEFORE-LPP: cir.global external @b = ctor : !rec_B { |
| // CIR-BEFORE-LPP: %[[OBJ:.*]] = cir.get_global @b : !cir.ptr<!rec_B> |
| // CIR-BEFORE-LPP: cir.call @_ZN1BC1Ev(%[[OBJ]]) : (!cir.ptr<!rec_B> {{.*}}) -> () |
| // CIR-BEFORE-LPP: } |
| |
| // Test case 'c' - before LoweringPrepare (internal linkage) |
| // CIR-BEFORE-LPP: cir.global {{.*}} internal {{.*}} @_ZL1c = ctor : !rec_C { |
| // CIR-BEFORE-LPP: %[[OBJ:.*]] = cir.get_global @_ZL1c : !cir.ptr<!rec_C> |
| // CIR-BEFORE-LPP: cir.call @_ZN1CC1Ev(%[[OBJ]]) : (!cir.ptr<!rec_C> {{.*}}) -> () |
| // CIR-BEFORE-LPP: %[[OBJ2:.*]] = cir.get_global @_ZL1c : !cir.ptr<!rec_C> |
| // CIR-BEFORE-LPP: } |
| |
| // Check all globals first (they appear at the top of LLVM/OGCG output) |
| // LLVM: @a ={{.*}} global {{.*}} zeroinitializer |
| // LLVM: @a2 ={{.*}} global {{.*}} zeroinitializer |
| // LLVM: @b ={{.*}} global {{.*}} zeroinitializer |
| // LLVM: @_ZL1c ={{.*}} global {{.*}} zeroinitializer |
| |
| // OGCG: @a ={{.*}} global {{.*}} zeroinitializer |
| // OGCG: @a2 ={{.*}} global {{.*}} zeroinitializer |
| // OGCG: @b ={{.*}} global {{.*}} zeroinitializer |
| // OGCG: @_ZL1c ={{.*}} global {{.*}} zeroinitializer |
| |
| // Test case 'a' - should have constant storage |
| // CIR checks for 'a' |
| // CIR: cir.global external @a = #cir.zero : !rec_A |
| // CIR: cir.func internal private @__cxx_global_var_init() { |
| // CIR: %[[OBJ:.*]] = cir.get_global @a : !cir.ptr<!rec_A> |
| // CIR: cir.call @_ZN1AC1Ev(%[[OBJ]]) : (!cir.ptr<!rec_A> {{.*}}) -> () |
| // CIR: cir.return |
| // CIR: } |
| |
| // LLVM checks for 'a' (no optimization) |
| // LLVM: define internal void @__cxx_global_var_init() { |
| // LLVM: call void @_ZN1AC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a) |
| // LLVM: ret void |
| // LLVM: } |
| |
| // OGCG checks for 'a' (no optimization) |
| // OGCG: define internal void @__cxx_global_var_init() {{.*}} section ".text.startup" { |
| // OGCG: call void @_ZN1AC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a) |
| // OGCG: ret void |
| // OGCG: } |
| |
| // Test case 'a2' - should have constant storage (constexpr dtor) |
| // CIR checks for 'a2' |
| // CIR: cir.global external @a2 = #cir.zero : !rec_A2 |
| // CIR: cir.func internal private @__cxx_global_var_init.1() { |
| // CIR: %[[OBJ:.*]] = cir.get_global @a2 : !cir.ptr<!rec_A2> |
| // CIR: cir.call @_ZN2A2C1Ev(%[[OBJ]]) : (!cir.ptr<!rec_A2> {{.*}}) -> () |
| // CIR: cir.return |
| // CIR: } |
| |
| // LLVM checks for 'a2' (no optimization) |
| // LLVM: define internal void @__cxx_global_var_init.1() { |
| // LLVM: call void @_ZN2A2C1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a2) |
| // LLVM: ret void |
| // LLVM: } |
| |
| // OGCG checks for 'a2' (no optimization) |
| // OGCG: define internal void @__cxx_global_var_init.1() {{.*}} section ".text.startup" { |
| // OGCG: call void @_ZN2A2C1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a2) |
| // OGCG: ret void |
| // OGCG: } |
| |
| // Test case 'b' - should NOT have constant storage (mutable member) |
| // CIR checks for 'b' |
| // CIR: cir.global external @b = #cir.zero : !rec_B |
| // CIR: cir.func internal private @__cxx_global_var_init.2() { |
| // CIR: %[[OBJ:.*]] = cir.get_global @b : !cir.ptr<!rec_B> |
| // CIR: cir.call @_ZN1BC1Ev(%[[OBJ]]) : (!cir.ptr<!rec_B> {{.*}}) -> () |
| // CIR: cir.return |
| // CIR: } |
| |
| // LLVM checks for 'b' (no optimization) |
| // LLVM: define internal void @__cxx_global_var_init.2() { |
| // LLVM: call void @_ZN1BC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @b) |
| // LLVM: ret void |
| // LLVM: } |
| |
| // OGCG checks for 'b' (no optimization) |
| // OGCG: define internal void @__cxx_global_var_init.2() {{.*}} section ".text.startup" { |
| // OGCG: call void @_ZN1BC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @b) |
| // OGCG: ret void |
| // OGCG: } |
| |
| // Test case 'c' - Andy's simple case, should have constant storage (internal linkage) |
| // CIR checks for 'c' |
| // CIR: cir.global {{.*}} internal {{.*}} @_ZL1c = #cir.zero : !rec_C |
| // CIR: cir.func internal private @__cxx_global_var_init.3() { |
| // CIR: %[[OBJ:.*]] = cir.get_global @_ZL1c : !cir.ptr<!rec_C> |
| // CIR: cir.call @_ZN1CC1Ev(%[[OBJ]]) : (!cir.ptr<!rec_C> {{.*}}) -> () |
| // CIR: cir.return |
| // CIR: } |
| |
| // LLVM checks for 'c' (no optimization) |
| // LLVM: define internal void @__cxx_global_var_init.3() { |
| // LLVM: call void @_ZN1CC1Ev(ptr noundef nonnull align 4 dereferenceable(8) @_ZL1c) |
| // LLVM: ret void |
| // LLVM: } |
| |
| // OGCG checks for 'c' (no optimization) |
| // OGCG: define internal void @__cxx_global_var_init.3() {{.*}} section ".text.startup" { |
| // OGCG: call void @_ZN1CC1Ev(ptr noundef nonnull align 4 dereferenceable(8) @_ZL1c) |
| // OGCG: ret void |
| // OGCG: } |
| |
| // With optimization enabled, should emit invariant.start intrinsic for constant storage cases |
| |
| // Check all globals first (they appear at the top of optimized LLVM/OGCG output) |
| // LLVM-OPT: @a ={{.*}} global {{.*}} zeroinitializer |
| // LLVM-OPT: @a2 ={{.*}} global {{.*}} zeroinitializer |
| // LLVM-OPT: @b ={{.*}} global {{.*}} zeroinitializer |
| // LLVM-OPT: @_ZL1c ={{.*}} global {{.*}} zeroinitializer |
| |
| // OGCG-OPT: @a ={{.*}} global {{.*}} zeroinitializer |
| // OGCG-OPT: @a2 ={{.*}} global {{.*}} zeroinitializer |
| // OGCG-OPT: @b ={{.*}} global {{.*}} zeroinitializer |
| // OGCG-OPT: @_ZL1c ={{.*}} global {{.*}} zeroinitializer |
| |
| // Test case 'a' - optimized checks |
| // LLVM-OPT: define internal void @__cxx_global_var_init() { |
| // LLVM-OPT: call void @_ZN1AC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a) |
| // LLVM-OPT: call {{.*}}@llvm.invariant.start.p0(i64 4, ptr @a) |
| // LLVM-OPT: ret void |
| // LLVM-OPT: } |
| |
| // OGCG-OPT: define internal void @__cxx_global_var_init() {{.*}} section ".text.startup" { |
| // OGCG-OPT: call void @_ZN1AC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a) |
| // OGCG-OPT: call {{.*}}@llvm.invariant.start.p0(i64 4, ptr @a) |
| // OGCG-OPT: ret void |
| // OGCG-OPT: } |
| |
| // Test case 'a2' - optimized checks |
| // LLVM-OPT: define internal void @__cxx_global_var_init.1() { |
| // LLVM-OPT: call void @_ZN2A2C1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a2) |
| // LLVM-OPT: call {{.*}}@llvm.invariant.start.p0(i64 4, ptr @a2) |
| // LLVM-OPT: ret void |
| // LLVM-OPT: } |
| |
| // OGCG-OPT: define internal void @__cxx_global_var_init.1() {{.*}} section ".text.startup" { |
| // OGCG-OPT: call void @_ZN2A2C1Ev(ptr noundef nonnull align 4 dereferenceable(4) @a2) |
| // OGCG-OPT: call {{.*}}@llvm.invariant.start.p0(i64 4, ptr @a2) |
| // OGCG-OPT: ret void |
| // OGCG-OPT: } |
| |
| // Test case 'b' - optimized checks (should NOT emit invariant.start) |
| // LLVM-OPT: define internal void @__cxx_global_var_init.2() { |
| // LLVM-OPT: call void @_ZN1BC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @b) |
| // LLVM-OPT-NOT: call {{.*}}@llvm.invariant.start.p0(i64 {{.*}}, ptr @b) |
| // LLVM-OPT: ret void |
| // LLVM-OPT: } |
| |
| // OGCG-OPT: define internal void @__cxx_global_var_init.2() {{.*}} section ".text.startup" { |
| // OGCG-OPT: call void @_ZN1BC1Ev(ptr noundef nonnull align 4 dereferenceable(4) @b) |
| // OGCG-OPT-NOT: call {{.*}}@llvm.invariant.start.p0(i64 {{.*}}, ptr @b) |
| // OGCG-OPT: ret void |
| // OGCG-OPT: } |
| |
| // Test case 'c' - optimized checks |
| // LLVM-OPT: define internal void @__cxx_global_var_init.3() { |
| // LLVM-OPT: call void @_ZN1CC1Ev(ptr noundef nonnull align 4 dereferenceable(8) @_ZL1c) |
| // LLVM-OPT: call {{.*}}@llvm.invariant.start.p0(i64 8, ptr @_ZL1c) |
| // LLVM-OPT: ret void |
| // LLVM-OPT: } |
| |
| // OGCG-OPT: define internal void @__cxx_global_var_init.3() {{.*}} section ".text.startup" { |
| // OGCG-OPT: call void @_ZN1CC1Ev(ptr noundef nonnull align 4 dereferenceable(8) @_ZL1c) |
| // OGCG-OPT: call {{.*}}@llvm.invariant.start.p0(i64 8, ptr @_ZL1c) |
| // OGCG-OPT: ret void |
| // OGCG-OPT: } |
| |
| // CIR checks at -O1 - should include invariant.start intrinsic calls for constant storage cases |
| // CIR-O1: module {{.*}} attributes {{.*}} cir.opt_info = #cir.opt_info<level = 1 |
| |
| // Test case 'a' - CIR at -O1 |
| // CIR-O1: cir.func internal private @__cxx_global_var_init() { |
| // CIR-O1: cir.call @_ZN1AC1Ev(%{{.*}}) : (!cir.ptr<!rec_A> {{.*}}) -> () |
| // CIR-O1: cir.call_llvm_intrinsic "invariant.start" {{.*}} : (!s64i, !cir.ptr<!rec_A>) -> !cir.ptr<!rec_A> |
| // CIR-O1: } |
| |
| // Test case 'a2' - CIR at -O1 |
| // CIR-O1: cir.func internal private @__cxx_global_var_init.1() { |
| // CIR-O1: cir.call @_ZN2A2C1Ev(%{{.*}}) : (!cir.ptr<!rec_A2> {{.*}}) -> () |
| // CIR-O1: cir.call_llvm_intrinsic "invariant.start" {{.*}} : (!s64i, !cir.ptr<!rec_A2>) -> !cir.ptr<!rec_A2> |
| // CIR-O1: } |
| |
| // Test case 'b' - CIR at -O1 (should NOT emit invariant.start) |
| // CIR-O1: cir.func internal private @__cxx_global_var_init.2() { |
| // CIR-O1: cir.call @_ZN1BC1Ev(%{{.*}}) : (!cir.ptr<!rec_B> {{.*}}) -> () |
| // CIR-O1-NOT: cir.call_llvm_intrinsic "invariant.start" |
| // CIR-O1: } |
| |
| // Test case 'c' - CIR at -O1 |
| // CIR-O1: cir.func internal private @__cxx_global_var_init.3() { |
| // CIR-O1: cir.call @_ZN1CC1Ev(%{{.*}}) : (!cir.ptr<!rec_C> {{.*}}) -> () |
| // CIR-O1: cir.call_llvm_intrinsic "invariant.start" {{.*}} : (!s64i, !cir.ptr<!rec_C>) -> !cir.ptr<!rec_C> |
| // CIR-O1: } |