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