1 // RUN: %clangxx_scudo -std=c++1z -faligned-allocation %s -o %t
2 // RUN: %run %t valid 2>&1
3 // RUN: %env_scudo_opts=allocator_may_return_null=1 %run %t invalid 2>&1
4 // RUN: %env_scudo_opts=allocator_may_return_null=0 not %run %t invalid 2>&1 | FileCheck %s
6 // Tests that the C++17 aligned new/delete operators are working as expected.
7 // Currently we do not check the consistency of the alignment on deallocation,
8 // so this just tests that the APIs work.
15 // Define all new/delete to not depend on the version provided by the platform.
19 static const nothrow_t nothrow
;
20 enum class align_val_t
: size_t {};
23 void *operator new(size_t);
24 void *operator new[](size_t);
25 void *operator new(size_t, std::nothrow_t
const &);
26 void *operator new[](size_t, std::nothrow_t
const &);
27 void *operator new(size_t, std::align_val_t
);
28 void *operator new[](size_t, std::align_val_t
);
29 void *operator new(size_t, std::align_val_t
, std::nothrow_t
const &);
30 void *operator new[](size_t, std::align_val_t
, std::nothrow_t
const &);
32 void operator delete(void *) throw();
33 void operator delete[](void *) throw();
34 void operator delete(void *, std::nothrow_t
const &);
35 void operator delete[](void *, std::nothrow_t
const &);
36 void operator delete(void *, size_t) throw();
37 void operator delete[](void *, size_t) throw();
38 void operator delete(void *, std::align_val_t
) throw();
39 void operator delete[](void *, std::align_val_t
) throw();
40 void operator delete(void *, std::align_val_t
, std::nothrow_t
const &);
41 void operator delete[](void *, std::align_val_t
, std::nothrow_t
const &);
42 void operator delete(void *, size_t, std::align_val_t
) throw();
43 void operator delete[](void *, size_t, std::align_val_t
) throw();
46 inline T
*break_optimization(T
*arg
) {
47 __asm__
__volatile__(""
57 struct alignas(128) S12_128
{
60 struct alignas(256) S12_256
{
63 struct alignas(512) S1024_512
{
66 struct alignas(1024) S1024_1024
{
70 int main(int argc
, char **argv
) {
73 if (!strcmp(argv
[1], "valid")) {
75 delete break_optimization(new S12
);
76 delete break_optimization(new S12_128
);
77 delete[] break_optimization(new S12_128
[4]);
78 delete break_optimization(new S12_256
);
79 delete break_optimization(new S1024_512
);
80 delete[] break_optimization(new S1024_512
[4]);
81 delete break_optimization(new S1024_1024
);
83 // Call directly the aligned versions of the operators.
84 const size_t alignment
= 1U << 8;
85 void *p
= operator new(1, static_cast<std::align_val_t
>(alignment
));
86 assert((reinterpret_cast<uintptr_t>(p
) & (alignment
- 1)) == 0);
87 operator delete(p
, static_cast<std::align_val_t
>(alignment
));
89 if (!strcmp(argv
[1], "invalid")) {
90 // Alignment must be a power of 2.
91 const size_t alignment
= (1U << 8) - 1;
92 void *p
= operator new(1, static_cast<std::align_val_t
>(alignment
),
94 // CHECK: Scudo ERROR: invalid allocation alignment