1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
|
// RUN: %clang_cc1 -std=c++20 -verify %s
using size_t = decltype(sizeof(0));
namespace std {
enum class align_val_t : size_t {};
struct destroying_delete_t {
explicit destroying_delete_t() = default;
};
inline constexpr destroying_delete_t destroying_delete{};
}
// Aligned version is preferred over unaligned version,
// unsized version is preferred over sized version.
template<unsigned Align>
struct alignas(Align) A {
void operator delete(void*);
void operator delete(void*, std::align_val_t) = delete; // expected-note {{here}}
void operator delete(void*, size_t) = delete;
void operator delete(void*, size_t, std::align_val_t) = delete;
};
void f(A<__STDCPP_DEFAULT_NEW_ALIGNMENT__> *p) { delete p; }
void f(A<__STDCPP_DEFAULT_NEW_ALIGNMENT__ * 2> *p) { delete p; } // expected-error {{deleted}}
template<unsigned Align>
struct alignas(Align) B {
void operator delete(void*, size_t);
void operator delete(void*, size_t, std::align_val_t) = delete; // expected-note {{here}}
};
void f(B<__STDCPP_DEFAULT_NEW_ALIGNMENT__> *p) { delete p; }
void f(B<__STDCPP_DEFAULT_NEW_ALIGNMENT__ * 2> *p) { delete p; } // expected-error {{deleted}}
// Ensure that a deleted destructor is acceptable when the selected overload
// for operator delete is a destroying delete. See the comments in GH118660.
struct S {
~S() = delete;
void operator delete(S *, std::destroying_delete_t) noexcept {}
};
struct T {
void operator delete(T *, std::destroying_delete_t) noexcept {}
private:
~T();
};
void foo(S *s, T *t) {
delete s; // Was rejected, is intended to be accepted.
delete t; // Was rejected, is intended to be accepted.
}
// However, if the destructor is virtual, then it has to be accessible because
// the behavior depends on which operator delete is selected and that is based
// on the dynamic type of the pointer.
struct U {
virtual ~U() = delete; // expected-note {{here}}
void operator delete(U *, std::destroying_delete_t) noexcept {}
};
struct V {
void operator delete(V *, std::destroying_delete_t) noexcept {}
private:
virtual ~V(); // expected-note {{here}}
};
void bar(U *u, V *v) {
// Both should be rejected because they have virtual destructors.
delete u; // expected-error {{attempt to use a deleted function}}
delete v; // expected-error {{calling a private destructor of class 'V'}}
}
|