xref: /aosp_15_r20/external/clang/test/CodeGenCXX/cxx0x-initializer-stdinitializerlist-startend.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 -std=c++11 -S -triple x86_64-none-linux-gnu -emit-llvm -o - %s | FileCheck %s
2*67e74705SXin Li 
3*67e74705SXin Li namespace std {
4*67e74705SXin Li   typedef decltype(sizeof(int)) size_t;
5*67e74705SXin Li 
6*67e74705SXin Li   // libc++'s implementation with __size_ replaced by __end_
7*67e74705SXin Li   template <class _E>
8*67e74705SXin Li   class initializer_list
9*67e74705SXin Li   {
10*67e74705SXin Li     const _E* __begin_;
11*67e74705SXin Li     const _E* __end_;
12*67e74705SXin Li 
initializer_list(const _E * __b,const _E * __e)13*67e74705SXin Li     initializer_list(const _E* __b, const _E* __e)
14*67e74705SXin Li       : __begin_(__b),
15*67e74705SXin Li         __end_(__e)
16*67e74705SXin Li     {}
17*67e74705SXin Li 
18*67e74705SXin Li   public:
19*67e74705SXin Li     typedef _E        value_type;
20*67e74705SXin Li     typedef const _E& reference;
21*67e74705SXin Li     typedef const _E& const_reference;
22*67e74705SXin Li     typedef size_t    size_type;
23*67e74705SXin Li 
24*67e74705SXin Li     typedef const _E* iterator;
25*67e74705SXin Li     typedef const _E* const_iterator;
26*67e74705SXin Li 
initializer_list()27*67e74705SXin Li     initializer_list() : __begin_(nullptr), __end_(nullptr) {}
28*67e74705SXin Li 
size() const29*67e74705SXin Li     size_t    size()  const {return __end_ - __begin_;}
begin() const30*67e74705SXin Li     const _E* begin() const {return __begin_;}
end() const31*67e74705SXin Li     const _E* end()   const {return __end_;}
32*67e74705SXin Li   };
33*67e74705SXin Li }
34*67e74705SXin Li 
35*67e74705SXin Li // CHECK: @_ZGR15globalInitList1_ = internal constant [3 x i32] [i32 1, i32 2, i32 3]
36*67e74705SXin Li // CHECK: @globalInitList1 = global {{[^ ]+}} { i32* getelementptr inbounds ([3 x i32], [3 x i32]* @_ZGR15globalInitList1_, {{[^)]*}}), i32*
37*67e74705SXin Li std::initializer_list<int> globalInitList1 = {1, 2, 3};
38*67e74705SXin Li 
fn1(int i)39*67e74705SXin Li void fn1(int i) {
40*67e74705SXin Li   // CHECK-LABEL: define void @_Z3fn1i
41*67e74705SXin Li   // temporary array
42*67e74705SXin Li   // CHECK: [[array:%[^ ]+]] = alloca [3 x i32]
43*67e74705SXin Li   // CHECK: getelementptr inbounds [3 x i32], [3 x i32]* [[array]], i{{32|64}} 0
44*67e74705SXin Li   // CHECK-NEXT: store i32 1, i32*
45*67e74705SXin Li   // CHECK-NEXT: getelementptr
46*67e74705SXin Li   // CHECK-NEXT: store
47*67e74705SXin Li   // CHECK-NEXT: getelementptr
48*67e74705SXin Li   // CHECK-NEXT: load
49*67e74705SXin Li   // CHECK-NEXT: store
50*67e74705SXin Li   // init the list
51*67e74705SXin Li   // CHECK-NEXT: getelementptr
52*67e74705SXin Li   // CHECK-NEXT: getelementptr inbounds [3 x i32], [3 x i32]*
53*67e74705SXin Li   // CHECK-NEXT: store i32*
54*67e74705SXin Li   // CHECK-NEXT: getelementptr
55*67e74705SXin Li   // CHECK-NEXT: getelementptr inbounds [3 x i32], [3 x i32]* [[array]], i{{32|64}} 0, i{{32|64}} 3
56*67e74705SXin Li   // CHECK-NEXT: store i32*
57*67e74705SXin Li   std::initializer_list<int> intlist{1, 2, i};
58*67e74705SXin Li }
59*67e74705SXin Li 
60*67e74705SXin Li struct destroyme1 {
61*67e74705SXin Li   ~destroyme1();
62*67e74705SXin Li };
63*67e74705SXin Li struct destroyme2 {
64*67e74705SXin Li   ~destroyme2();
65*67e74705SXin Li };
66*67e74705SXin Li 
67*67e74705SXin Li 
fn2()68*67e74705SXin Li void fn2() {
69*67e74705SXin Li   // CHECK-LABEL: define void @_Z3fn2v
70*67e74705SXin Li   void target(std::initializer_list<destroyme1>);
71*67e74705SXin Li   // objects should be destroyed before dm2, after call returns
72*67e74705SXin Li   target({ destroyme1(), destroyme1() });
73*67e74705SXin Li   // CHECK: call void @_ZN10destroyme1D1Ev
74*67e74705SXin Li   destroyme2 dm2;
75*67e74705SXin Li   // CHECK: call void @_ZN10destroyme2D1Ev
76*67e74705SXin Li }
77*67e74705SXin Li 
fn3()78*67e74705SXin Li void fn3() {
79*67e74705SXin Li   // CHECK-LABEL: define void @_Z3fn3v
80*67e74705SXin Li   // objects should be destroyed after dm2
81*67e74705SXin Li   auto list = { destroyme1(), destroyme1() };
82*67e74705SXin Li   destroyme2 dm2;
83*67e74705SXin Li   // CHECK: call void @_ZN10destroyme2D1Ev
84*67e74705SXin Li   // CHECK: call void @_ZN10destroyme1D1Ev
85*67e74705SXin Li }
86