blob: 4098beb56376e3552da4516b33cc122c7488ffba [file] [log] [blame]
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fsized-deallocation -faligned-allocation
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fno-sized-deallocation -faligned-allocation
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fsized-deallocation -fno-aligned-allocation
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fno-sized-deallocation -fno-aligned-allocation
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fsized-deallocation -faligned-allocation -fexperimental-new-constant-interpreter
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fno-sized-deallocation -faligned-allocation -fexperimental-new-constant-interpreter
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fsized-deallocation -fno-aligned-allocation -fexperimental-new-constant-interpreter
// RUN: %clang_cc1 -fsyntax-only -verify %s -std=c++26 -Wno-ext-cxx-type-aware-allocators -fexceptions -fno-sized-deallocation -fno-aligned-allocation -fexperimental-new-constant-interpreter
namespace std {
template <class T> struct type_identity {};
enum class align_val_t : __SIZE_TYPE__ {};
struct destroying_delete_t { explicit destroying_delete_t() = default; };
}
using size_t = __SIZE_TYPE__;
struct S1 {
constexpr explicit S1() : i(5) { }
const int i;
};
void *operator new(std::type_identity<S1>, size_t sz, std::align_val_t); // #1
void operator delete(std::type_identity<S1>, void* ptr, size_t sz, std::align_val_t); // #2
constexpr int ensure_consteval_skips_typed_allocators() {
// Verify we dont resolve typed allocators in const contexts
auto * s = new S1();
auto result = s->i;
delete s;
return result;
};
struct S2 {
constexpr explicit S2() : i(5) { }
const int i;
};
void *operator new(std::type_identity<S2>, size_t sz, std::align_val_t) = delete; // #3
void operator delete(std::type_identity<S2>, void* ptr, size_t sz, std::align_val_t) = delete; // #4
constexpr int ensure_constexpr_retains_types_at_runtime() {
// Verify we dont resolve typed allocators in const contexts
S2 *s = new S2();
// expected-error@-1 {{call to deleted function 'operator new'}}
// expected-note@#1 {{candidate function not viable: no known conversion from 'type_identity<S2>' to 'type_identity<S1>' for 1st argument}}
// expected-note@#3 {{candidate function has been explicitly deleted}}
auto result = s->i;
delete s;
// expected-error@-1 {{attempt to use a deleted function}}
// expected-note@#4 {{'operator delete' has been explicitly marked deleted here}}
return result;
};
struct S3 {
constexpr explicit S3() : i(5) { }
const int i;
template <typename T> void* operator new(std::type_identity<T>, size_t sz, std::align_val_t) = delete; // #5
template <typename T> void operator delete(std::type_identity<T>, void *, size_t sz, std::align_val_t) = delete; // #6
};
template <typename T> void* operator new(std::type_identity<T>, size_t sz, std::align_val_t) = delete; // #7
template <typename T> void operator delete(std::type_identity<T>, void *, size_t sz, std::align_val_t) = delete; // #8
constexpr int constexpr_vs_inclass_operators() {
S3 *s;
if consteval {
s = ::new S3();
// expected-error@-1 {{call to deleted function 'operator new'}}
// expected-note@#1 {{candidate function not viable: no known conversion from 'type_identity<S3>' to 'type_identity<S1>' for 1st argument}}
// expected-note@#3 {{candidate function not viable: no known conversion from 'type_identity<S3>' to 'type_identity<S2>' for 1st argument}}
// expected-note@#7 {{candidate function [with T = S3] has been explicitly deleted}}
} else {
s = new S3();
// expected-error@-1 {{call to deleted function 'operator new'}}
// expected-note@#5 {{candidate function [with T = S3] has been explicitly deleted}}
}
auto result = s->i;
if consteval {
::delete s;
// expected-error@-1 {{attempt to use a deleted function}}
// expected-note@#8 {{'operator delete<S3>' has been explicitly marked deleted here}}
} else {
delete s;
// expected-error@-1 {{attempt to use a deleted function}}
// expected-note@#6 {{'operator delete<S3>' has been explicitly marked deleted here}}
}
return result;
};
// Test a variety of valid constant evaluation paths
struct S4 {
int i = 1;
constexpr S4() __attribute__((noinline)) {}
};
void* operator new(std::type_identity<S4>, size_t sz, std::align_val_t);
void operator delete(std::type_identity<S4>, void *, size_t sz, std::align_val_t);
constexpr int do_dynamic_alloc(int n) {
S4* s = new S4;
int result = n * s->i;
delete s;
return result;
}
template <int N> struct Tag {
};
static constexpr int force_do_dynamic_alloc = do_dynamic_alloc(5);
constexpr int test_consteval_calling_constexpr(int i) {
if consteval {
return do_dynamic_alloc(2 * i);
}
return do_dynamic_alloc(3 * i);
}
int test_consteval(int n, Tag<test_consteval_calling_constexpr(2)>, Tag<do_dynamic_alloc(3)>) {
static const int t1 = test_consteval_calling_constexpr(4);
static const int t2 = do_dynamic_alloc(5);
int t3 = test_consteval_calling_constexpr(6);
int t4 = do_dynamic_alloc(7);
return t1 * t2 * t3 * t4;
}