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'}}
}
|