xref: /aosp_15_r20/external/clang/test/CodeGenCXX/new.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 -triple x86_64-unknown-unknown %s -emit-llvm -o - | FileCheck %s
2*67e74705SXin Li 
3*67e74705SXin Li typedef __typeof__(sizeof(0)) size_t;
4*67e74705SXin Li 
5*67e74705SXin Li // Declare an 'operator new' template to tickle a bug in __builtin_operator_new.
6*67e74705SXin Li template<typename T> void *operator new(size_t, int (*)(T));
7*67e74705SXin Li 
8*67e74705SXin Li // Ensure that this declaration doesn't cause operator new to lose its
9*67e74705SXin Li // 'noalias' attribute.
10*67e74705SXin Li void *operator new[](size_t);
11*67e74705SXin Li 
t1()12*67e74705SXin Li void t1() {
13*67e74705SXin Li   delete new int;
14*67e74705SXin Li   delete [] new int [3];
15*67e74705SXin Li }
16*67e74705SXin Li 
17*67e74705SXin Li // CHECK: declare noalias i8* @_Znwm(i64) [[ATTR_NOBUILTIN:#[^ ]*]]
18*67e74705SXin Li // CHECK: declare void @_ZdlPv(i8*) [[ATTR_NOBUILTIN_NOUNWIND:#[^ ]*]]
19*67e74705SXin Li // CHECK: declare noalias i8* @_Znam(i64) [[ATTR_NOBUILTIN]]
20*67e74705SXin Li // CHECK: declare void @_ZdaPv(i8*) [[ATTR_NOBUILTIN_NOUNWIND]]
21*67e74705SXin Li 
22*67e74705SXin Li namespace std {
23*67e74705SXin Li   struct nothrow_t {};
24*67e74705SXin Li }
25*67e74705SXin Li std::nothrow_t nothrow;
26*67e74705SXin Li 
27*67e74705SXin Li // Declare the reserved placement operators.
28*67e74705SXin Li void *operator new(size_t, void*) throw();
29*67e74705SXin Li void operator delete(void*, void*) throw();
30*67e74705SXin Li void *operator new[](size_t, void*) throw();
31*67e74705SXin Li void operator delete[](void*, void*) throw();
32*67e74705SXin Li 
33*67e74705SXin Li // Declare the replaceable global allocation operators.
34*67e74705SXin Li void *operator new(size_t, const std::nothrow_t &) throw();
35*67e74705SXin Li void *operator new[](size_t, const std::nothrow_t &) throw();
36*67e74705SXin Li void operator delete(void *, const std::nothrow_t &) throw();
37*67e74705SXin Li void operator delete[](void *, const std::nothrow_t &) throw();
38*67e74705SXin Li 
39*67e74705SXin Li // Declare some other placemenet operators.
40*67e74705SXin Li void *operator new(size_t, void*, bool) throw();
41*67e74705SXin Li void *operator new[](size_t, void*, bool) throw();
42*67e74705SXin Li 
t2(int * a)43*67e74705SXin Li void t2(int* a) {
44*67e74705SXin Li   int* b = new (a) int;
45*67e74705SXin Li }
46*67e74705SXin Li 
47*67e74705SXin Li struct S {
48*67e74705SXin Li   int a;
49*67e74705SXin Li };
50*67e74705SXin Li 
51*67e74705SXin Li // POD types.
t3()52*67e74705SXin Li void t3() {
53*67e74705SXin Li   int *a = new int(10);
54*67e74705SXin Li   _Complex int* b = new _Complex int(10i);
55*67e74705SXin Li 
56*67e74705SXin Li   S s;
57*67e74705SXin Li   s.a = 10;
58*67e74705SXin Li   S *sp = new S(s);
59*67e74705SXin Li }
60*67e74705SXin Li 
61*67e74705SXin Li // Non-POD
62*67e74705SXin Li struct T {
63*67e74705SXin Li   T();
64*67e74705SXin Li   int a;
65*67e74705SXin Li };
66*67e74705SXin Li 
t4()67*67e74705SXin Li void t4() {
68*67e74705SXin Li   // CHECK: call void @_ZN1TC1Ev
69*67e74705SXin Li   T *t = new T;
70*67e74705SXin Li }
71*67e74705SXin Li 
72*67e74705SXin Li struct T2 {
73*67e74705SXin Li   int a;
74*67e74705SXin Li   T2(int, int);
75*67e74705SXin Li };
76*67e74705SXin Li 
t5()77*67e74705SXin Li void t5() {
78*67e74705SXin Li   // CHECK: call void @_ZN2T2C1Eii
79*67e74705SXin Li   T2 *t2 = new T2(10, 10);
80*67e74705SXin Li }
81*67e74705SXin Li 
t6()82*67e74705SXin Li int *t6() {
83*67e74705SXin Li   // Null check.
84*67e74705SXin Li   return new (0) int(10);
85*67e74705SXin Li }
86*67e74705SXin Li 
t7()87*67e74705SXin Li void t7() {
88*67e74705SXin Li   new int();
89*67e74705SXin Li }
90*67e74705SXin Li 
91*67e74705SXin Li struct U {
92*67e74705SXin Li   ~U();
93*67e74705SXin Li };
94*67e74705SXin Li 
t8(int n)95*67e74705SXin Li void t8(int n) {
96*67e74705SXin Li   new int[10];
97*67e74705SXin Li   new int[n];
98*67e74705SXin Li 
99*67e74705SXin Li   // Non-POD
100*67e74705SXin Li   new T[10];
101*67e74705SXin Li   new T[n];
102*67e74705SXin Li 
103*67e74705SXin Li   // Cookie required
104*67e74705SXin Li   new U[10];
105*67e74705SXin Li   new U[n];
106*67e74705SXin Li }
107*67e74705SXin Li 
t9()108*67e74705SXin Li void t9() {
109*67e74705SXin Li   bool b;
110*67e74705SXin Li 
111*67e74705SXin Li   new bool(true);
112*67e74705SXin Li   new (&b) bool(true);
113*67e74705SXin Li }
114*67e74705SXin Li 
115*67e74705SXin Li struct A {
116*67e74705SXin Li   void* operator new(__typeof(sizeof(int)), int, float, ...);
117*67e74705SXin Li   A();
118*67e74705SXin Li };
119*67e74705SXin Li 
t10()120*67e74705SXin Li A* t10() {
121*67e74705SXin Li    // CHECK: @_ZN1AnwEmifz
122*67e74705SXin Li   return new(1, 2, 3.45, 100) A;
123*67e74705SXin Li }
124*67e74705SXin Li 
125*67e74705SXin Li // CHECK-LABEL: define void @_Z3t11i
126*67e74705SXin Li struct B { int a; };
127*67e74705SXin Li struct Bmemptr { int Bmemptr::* memptr; int a; };
128*67e74705SXin Li 
t11(int n)129*67e74705SXin Li void t11(int n) {
130*67e74705SXin Li   // CHECK: call i8* @_Znwm
131*67e74705SXin Li   // CHECK: call void @llvm.memset.p0i8.i64(
132*67e74705SXin Li   B* b = new B();
133*67e74705SXin Li 
134*67e74705SXin Li   // CHECK: call i8* @_Znam
135*67e74705SXin Li   // CHECK: {{call void.*llvm.memset.p0i8.i64.*i8 0, i64 %}}
136*67e74705SXin Li   B *b2 = new B[n]();
137*67e74705SXin Li 
138*67e74705SXin Li   // CHECK: call i8* @_Znam
139*67e74705SXin Li   // CHECK: call void @llvm.memcpy.p0i8.p0i8.i64
140*67e74705SXin Li   // CHECK: br
141*67e74705SXin Li   Bmemptr *b_memptr = new Bmemptr[n]();
142*67e74705SXin Li 
143*67e74705SXin Li   // CHECK: ret void
144*67e74705SXin Li }
145*67e74705SXin Li 
146*67e74705SXin Li struct Empty { };
147*67e74705SXin Li 
148*67e74705SXin Li // We don't need to initialize an empty class.
149*67e74705SXin Li // CHECK-LABEL: define void @_Z3t12v
t12()150*67e74705SXin Li void t12() {
151*67e74705SXin Li   // CHECK: call i8* @_Znam
152*67e74705SXin Li   // CHECK-NOT: br
153*67e74705SXin Li   (void)new Empty[10];
154*67e74705SXin Li 
155*67e74705SXin Li   // CHECK: call i8* @_Znam
156*67e74705SXin Li   // CHECK-NOT: br
157*67e74705SXin Li   (void)new Empty[10]();
158*67e74705SXin Li 
159*67e74705SXin Li   // CHECK: ret void
160*67e74705SXin Li }
161*67e74705SXin Li 
162*67e74705SXin Li // Zero-initialization
163*67e74705SXin Li // CHECK-LABEL: define void @_Z3t13i
t13(int n)164*67e74705SXin Li void t13(int n) {
165*67e74705SXin Li   // CHECK: call i8* @_Znwm
166*67e74705SXin Li   // CHECK: store i32 0, i32*
167*67e74705SXin Li   (void)new int();
168*67e74705SXin Li 
169*67e74705SXin Li   // CHECK: call i8* @_Znam
170*67e74705SXin Li   // CHECK: {{call void.*llvm.memset.p0i8.i64.*i8 0, i64 %}}
171*67e74705SXin Li   (void)new int[n]();
172*67e74705SXin Li 
173*67e74705SXin Li   // CHECK-NEXT: ret void
174*67e74705SXin Li }
175*67e74705SXin Li 
176*67e74705SXin Li struct Alloc{
177*67e74705SXin Li   int x;
178*67e74705SXin Li   void* operator new[](size_t size);
179*67e74705SXin Li   void operator delete[](void* p);
180*67e74705SXin Li   ~Alloc();
181*67e74705SXin Li };
182*67e74705SXin Li 
f()183*67e74705SXin Li void f() {
184*67e74705SXin Li   // CHECK: call i8* @_ZN5AllocnaEm(i64 808)
185*67e74705SXin Li   // CHECK: store i64 200
186*67e74705SXin Li   // CHECK: call void @_ZN5AllocD1Ev(
187*67e74705SXin Li   // CHECK: call void @_ZN5AllocdaEPv(i8*
188*67e74705SXin Li   delete[] new Alloc[10][20];
189*67e74705SXin Li   // CHECK: call i8* @_Znwm
190*67e74705SXin Li   // CHECK: call void @_ZdlPv(i8*
191*67e74705SXin Li   delete new bool;
192*67e74705SXin Li   // CHECK: ret void
193*67e74705SXin Li }
194*67e74705SXin Li 
195*67e74705SXin Li namespace test15 {
196*67e74705SXin Li   struct A { A(); ~A(); };
197*67e74705SXin Li 
198*67e74705SXin Li   // CHECK-LABEL:    define void @_ZN6test156test0aEPv(
199*67e74705SXin Li   // CHECK:      [[P:%.*]] = load i8*, i8**
200*67e74705SXin Li   // CHECK-NOT:  icmp eq i8* [[P]], null
201*67e74705SXin Li   // CHECK-NOT:  br i1
202*67e74705SXin Li   // CHECK:      [[T0:%.*]] = bitcast i8* [[P]] to [[A:%.*]]*
203*67e74705SXin Li   // CHECK-NEXT: call void @_ZN6test151AC1Ev([[A]]* [[T0]])
test0a(void * p)204*67e74705SXin Li   void test0a(void *p) {
205*67e74705SXin Li     new (p) A();
206*67e74705SXin Li   }
207*67e74705SXin Li 
208*67e74705SXin Li   // CHECK-LABEL:    define void @_ZN6test156test0bEPv(
209*67e74705SXin Li   // CHECK:      [[P0:%.*]] = load i8*, i8**
210*67e74705SXin Li   // CHECK:      [[P:%.*]] = call i8* @_ZnwmPvb(i64 1, i8* [[P0]]
211*67e74705SXin Li   // CHECK-NEXT: icmp eq i8* [[P]], null
212*67e74705SXin Li   // CHECK-NEXT: br i1
213*67e74705SXin Li   // CHECK:      [[T0:%.*]] = bitcast i8* [[P]] to [[A:%.*]]*
214*67e74705SXin Li   // CHECK-NEXT: call void @_ZN6test151AC1Ev([[A]]* [[T0]])
test0b(void * p)215*67e74705SXin Li   void test0b(void *p) {
216*67e74705SXin Li     new (p, true) A();
217*67e74705SXin Li   }
218*67e74705SXin Li 
219*67e74705SXin Li   // CHECK-LABEL:    define void @_ZN6test156test1aEPv(
220*67e74705SXin Li   // CHECK:      [[P:%.*]] = load i8*, i8**
221*67e74705SXin Li   // CHECK-NOT:  icmp eq i8* [[P]], null
222*67e74705SXin Li   // CHECK-NOT:  br i1
223*67e74705SXin Li   // CHECK:      [[BEGIN:%.*]] = bitcast i8* [[P]] to [[A:%.*]]*
224*67e74705SXin Li   // CHECK-NEXT: [[END:%.*]] = getelementptr inbounds [[A]], [[A]]* [[BEGIN]], i64 5
225*67e74705SXin Li   // CHECK-NEXT: br label
226*67e74705SXin Li   // CHECK:      [[CUR:%.*]] = phi [[A]]* [ [[BEGIN]], {{%.*}} ], [ [[NEXT:%.*]], {{%.*}} ]
227*67e74705SXin Li   // CHECK-NEXT: call void @_ZN6test151AC1Ev([[A]]* [[CUR]])
228*67e74705SXin Li   // CHECK-NEXT: [[NEXT]] = getelementptr inbounds [[A]], [[A]]* [[CUR]], i64 1
229*67e74705SXin Li   // CHECK-NEXT: [[DONE:%.*]] = icmp eq [[A]]* [[NEXT]], [[END]]
230*67e74705SXin Li   // CHECK-NEXT: br i1 [[DONE]]
test1a(void * p)231*67e74705SXin Li   void test1a(void *p) {
232*67e74705SXin Li     new (p) A[5];
233*67e74705SXin Li   }
234*67e74705SXin Li 
235*67e74705SXin Li   // CHECK-LABEL:    define void @_ZN6test156test1bEPv(
236*67e74705SXin Li   // CHECK:      [[P0:%.*]] = load i8*, i8**
237*67e74705SXin Li   // CHECK:      [[P:%.*]] = call i8* @_ZnamPvb(i64 13, i8* [[P0]]
238*67e74705SXin Li   // CHECK-NEXT: icmp eq i8* [[P]], null
239*67e74705SXin Li   // CHECK-NEXT: br i1
240*67e74705SXin Li   // CHECK:      [[AFTER_COOKIE:%.*]] = getelementptr inbounds i8, i8* [[P]], i64 8
241*67e74705SXin Li   // CHECK:      [[BEGIN:%.*]] = bitcast i8* [[AFTER_COOKIE]] to [[A:%.*]]*
242*67e74705SXin Li   // CHECK-NEXT: [[END:%.*]] = getelementptr inbounds [[A]], [[A]]* [[BEGIN]], i64 5
243*67e74705SXin Li   // CHECK-NEXT: br label
244*67e74705SXin Li   // CHECK:      [[CUR:%.*]] = phi [[A]]* [ [[BEGIN]], {{%.*}} ], [ [[NEXT:%.*]], {{%.*}} ]
245*67e74705SXin Li   // CHECK-NEXT: call void @_ZN6test151AC1Ev([[A]]* [[CUR]])
246*67e74705SXin Li   // CHECK-NEXT: [[NEXT]] = getelementptr inbounds [[A]], [[A]]* [[CUR]], i64 1
247*67e74705SXin Li   // CHECK-NEXT: [[DONE:%.*]] = icmp eq [[A]]* [[NEXT]], [[END]]
248*67e74705SXin Li   // CHECK-NEXT: br i1 [[DONE]]
test1b(void * p)249*67e74705SXin Li   void test1b(void *p) {
250*67e74705SXin Li     new (p, true) A[5];
251*67e74705SXin Li   }
252*67e74705SXin Li 
253*67e74705SXin Li   // TODO: it's okay if all these size calculations get dropped.
254*67e74705SXin Li   // FIXME: maybe we should try to throw on overflow?
255*67e74705SXin Li   // CHECK-LABEL:    define void @_ZN6test155test2EPvi(
256*67e74705SXin Li   // CHECK:      [[N:%.*]] = load i32, i32*
257*67e74705SXin Li   // CHECK-NEXT: [[T0:%.*]] = sext i32 [[N]] to i64
258*67e74705SXin Li   // CHECK-NEXT: [[T1:%.*]] = icmp slt i64 [[T0]], 0
259*67e74705SXin Li   // CHECK-NEXT: [[T2:%.*]] = select i1 [[T1]], i64 -1, i64 [[T0]]
260*67e74705SXin Li   // CHECK-NEXT: [[P:%.*]] = load i8*, i8**
261*67e74705SXin Li   // CHECK:      [[BEGIN:%.*]] = bitcast i8* [[P]] to [[A:%.*]]*
262*67e74705SXin Li   // CHECK-NEXT: [[ISEMPTY:%.*]] = icmp eq i64 [[T0]], 0
263*67e74705SXin Li   // CHECK-NEXT: br i1 [[ISEMPTY]],
264*67e74705SXin Li   // CHECK:      [[END:%.*]] = getelementptr inbounds [[A]], [[A]]* [[BEGIN]], i64 [[T0]]
265*67e74705SXin Li   // CHECK-NEXT: br label
266*67e74705SXin Li   // CHECK:      [[CUR:%.*]] = phi [[A]]* [ [[BEGIN]],
267*67e74705SXin Li   // CHECK-NEXT: call void @_ZN6test151AC1Ev([[A]]* [[CUR]])
test2(void * p,int n)268*67e74705SXin Li   void test2(void *p, int n) {
269*67e74705SXin Li     new (p) A[n];
270*67e74705SXin Li   }
271*67e74705SXin Li }
272*67e74705SXin Li 
273*67e74705SXin Li namespace PR10197 {
274*67e74705SXin Li   // CHECK-LABEL: define weak_odr void @_ZN7PR101971fIiEEvv()
275*67e74705SXin Li   template<typename T>
f()276*67e74705SXin Li   void f() {
277*67e74705SXin Li     // CHECK: [[CALL:%.*]] = call i8* @_Znwm
278*67e74705SXin Li     // CHECK-NEXT: [[CASTED:%.*]] = bitcast i8* [[CALL]] to
279*67e74705SXin Li     new T;
280*67e74705SXin Li     // CHECK-NEXT: ret void
281*67e74705SXin Li   }
282*67e74705SXin Li 
283*67e74705SXin Li   template void f<int>();
284*67e74705SXin Li }
285*67e74705SXin Li 
286*67e74705SXin Li namespace PR11523 {
287*67e74705SXin Li   class MyClass;
288*67e74705SXin Li   typedef int MyClass::* NewTy;
289*67e74705SXin Li   // CHECK-LABEL: define i64* @_ZN7PR115231fEv
290*67e74705SXin Li   // CHECK: store i64 -1
f()291*67e74705SXin Li   NewTy* f() { return new NewTy[2](); }
292*67e74705SXin Li }
293*67e74705SXin Li 
294*67e74705SXin Li namespace PR11757 {
295*67e74705SXin Li   // Make sure we elide the copy construction.
296*67e74705SXin Li   struct X { X(); X(const X&); };
a(X * x)297*67e74705SXin Li   X* a(X* x) { return new X(X()); }
298*67e74705SXin Li   // CHECK: define {{.*}} @_ZN7PR117571aEPNS_1XE
299*67e74705SXin Li   // CHECK: [[CALL:%.*]] = call i8* @_Znwm
300*67e74705SXin Li   // CHECK-NEXT: [[CASTED:%.*]] = bitcast i8* [[CALL]] to
301*67e74705SXin Li   // CHECK-NEXT: call void @_ZN7PR117571XC1Ev({{.*}}* [[CASTED]])
302*67e74705SXin Li   // CHECK-NEXT: ret {{.*}} [[CASTED]]
303*67e74705SXin Li }
304*67e74705SXin Li 
305*67e74705SXin Li namespace PR13380 {
APR13380::A306*67e74705SXin Li   struct A { A() {} };
307*67e74705SXin Li   struct B : public A { int x; };
308*67e74705SXin Li   // CHECK-LABEL: define i8* @_ZN7PR133801fEv
309*67e74705SXin Li   // CHECK: call i8* @_Znam(
310*67e74705SXin Li   // CHECK: call void @llvm.memset.p0i8
311*67e74705SXin Li   // CHECK-NEXT: call void @_ZN7PR133801BC1Ev
f()312*67e74705SXin Li   void* f() { return new B[2](); }
313*67e74705SXin Li }
314*67e74705SXin Li 
315*67e74705SXin Li struct MyPlacementType {} mpt;
316*67e74705SXin Li void *operator new(size_t, MyPlacementType);
317*67e74705SXin Li 
318*67e74705SXin Li namespace N3664 {
319*67e74705SXin Li   struct S { S() throw(int); };
320*67e74705SXin Li 
321*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5N36641fEv
f()322*67e74705SXin Li   void f() {
323*67e74705SXin Li     // CHECK: call i8* @_Znwm(i64 4) [[ATTR_BUILTIN_NEW:#[^ ]*]]
324*67e74705SXin Li     int *p = new int; // expected-note {{allocated with 'new' here}}
325*67e74705SXin Li     // CHECK: call void @_ZdlPv({{.*}}) [[ATTR_BUILTIN_DELETE:#[^ ]*]]
326*67e74705SXin Li     delete p;
327*67e74705SXin Li 
328*67e74705SXin Li     // CHECK: call i8* @_Znam(i64 12) [[ATTR_BUILTIN_NEW]]
329*67e74705SXin Li     int *q = new int[3];
330*67e74705SXin Li     // CHECK: call void @_ZdaPv({{.*}}) [[ATTR_BUILTIN_DELETE]]
331*67e74705SXin Li     delete[] p; // expected-warning {{'delete[]' applied to a pointer that was allocated with 'new'; did you mean 'delete'?}}
332*67e74705SXin Li 
333*67e74705SXin Li     // CHECK: call i8* @_ZnamRKSt9nothrow_t(i64 3, {{.*}}) [[ATTR_BUILTIN_NOTHROW_NEW:#[^ ]*]]
334*67e74705SXin Li     (void) new (nothrow) S[3];
335*67e74705SXin Li 
336*67e74705SXin Li     // CHECK: call i8* @_Znwm15MyPlacementType(i64 4){{$}}
337*67e74705SXin Li     (void) new (mpt) int;
338*67e74705SXin Li   }
339*67e74705SXin Li 
340*67e74705SXin Li   // CHECK: declare noalias i8* @_ZnamRKSt9nothrow_t(i64, {{.*}}) [[ATTR_NOBUILTIN_NOUNWIND]]
341*67e74705SXin Li 
342*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5N36641gEv
g()343*67e74705SXin Li   void g() {
344*67e74705SXin Li     // It's OK for there to be attributes here, so long as we don't have a
345*67e74705SXin Li     // 'builtin' attribute.
346*67e74705SXin Li     // CHECK: call i8* @_Znwm(i64 4){{$}}
347*67e74705SXin Li     int *p = (int*)operator new(4);
348*67e74705SXin Li     // CHECK: call void @_ZdlPv({{.*}}) [[ATTR_NOUNWIND:#[^ ]*]]
349*67e74705SXin Li     operator delete(p);
350*67e74705SXin Li 
351*67e74705SXin Li     // CHECK: call i8* @_Znam(i64 12){{$}}
352*67e74705SXin Li     int *q = (int*)operator new[](12);
353*67e74705SXin Li     // CHECK: call void @_ZdaPv({{.*}}) [[ATTR_NOUNWIND]]
354*67e74705SXin Li     operator delete [](p);
355*67e74705SXin Li 
356*67e74705SXin Li     // CHECK: call i8* @_ZnamRKSt9nothrow_t(i64 3, {{.*}}) [[ATTR_NOUNWIND]]
357*67e74705SXin Li     (void) operator new[](3, nothrow);
358*67e74705SXin Li   }
359*67e74705SXin Li }
360*67e74705SXin Li 
361*67e74705SXin Li namespace builtins {
362*67e74705SXin Li   // CHECK-LABEL: define void @_ZN8builtins1fEv
f()363*67e74705SXin Li   void f() {
364*67e74705SXin Li     // CHECK: call i8* @_Znwm(i64 4) [[ATTR_BUILTIN_NEW]]
365*67e74705SXin Li     // CHECK: call void @_ZdlPv({{.*}}) [[ATTR_BUILTIN_DELETE]]
366*67e74705SXin Li     __builtin_operator_delete(__builtin_operator_new(4));
367*67e74705SXin Li   }
368*67e74705SXin Li }
369*67e74705SXin Li 
370*67e74705SXin Li // CHECK-DAG: attributes [[ATTR_NOBUILTIN]] = {{[{].*}} nobuiltin {{.*[}]}}
371*67e74705SXin Li // CHECK-DAG: attributes [[ATTR_NOBUILTIN_NOUNWIND]] = {{[{].*}} nobuiltin nounwind {{.*[}]}}
372*67e74705SXin Li 
373*67e74705SXin Li // CHECK-DAG: attributes [[ATTR_BUILTIN_NEW]] = {{[{].*}} builtin {{.*[}]}}
374*67e74705SXin Li // CHECK-DAG: attributes [[ATTR_BUILTIN_DELETE]] = {{[{].*}} builtin {{.*[}]}}
375*67e74705SXin Li 
376*67e74705SXin Li // The ([^b}|...) monstrosity is matching a character that's not the start of 'builtin'.
377*67e74705SXin Li // Add more letters if this matches some other attribute.
378*67e74705SXin Li // CHECK-DAG: attributes [[ATTR_NOUNWIND]] = {{([^b]|b[^u]|bu[^i]|bui[^l])*}} nounwind {{([^b]|b[^u]|bu[^i]|bui[^l])*$}}
379