mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-16 17:46:40 +00:00

The PR reapply https://github.com/llvm/llvm-project/pull/97308. - Implement [CWG1815](https://wg21.link/CWG1815): Support lifetime extension of temporary created by aggregate initialization using a default member initializer. - Fix crash that introduced in https://github.com/llvm/llvm-project/pull/97308. In `InitListChecker::FillInEmptyInitForField`, when we enter rebuild-default-init context, we copy all the contents of the parent context to the current context, which will cause the `MaybeODRUseExprs` to be lost. But we don't need to copy the entire context, only the `DelayedDefaultInitializationContext` was required, which is used to build `SourceLocExpr`, etc. --------- Signed-off-by: yronglin <yronglin777@gmail.com>
208 lines
4.7 KiB
C++
208 lines
4.7 KiB
C++
// RUN: %clang_cc1 -std=c++11 -verify %s -pedantic
|
|
// RUN: %clang_cc1 -std=c++11 -verify %s -pedantic -fexperimental-new-constant-interpreter
|
|
// RUN: %clang_cc1 -std=c++20 -verify %s -pedantic
|
|
// RUN: %clang_cc1 -std=c++20 -verify %s -pedantic -fexperimental-new-constant-interpreter
|
|
|
|
|
|
namespace PR31692 {
|
|
struct A {
|
|
struct X { int n = 0; } x;
|
|
// Trigger construction of X() from a SFINAE context. This must not mark
|
|
// any part of X as invalid.
|
|
static_assert(!__is_constructible(X), "");
|
|
// Check that X::n is not marked invalid.
|
|
double &r = x.n; // expected-error {{non-const lvalue reference to type 'double' cannot bind to a value of unrelated type 'int'}}
|
|
};
|
|
// A::X can now be default-constructed.
|
|
static_assert(__is_constructible(A::X), "");
|
|
}
|
|
|
|
|
|
struct S {
|
|
} constexpr s;
|
|
struct C {
|
|
C(S);
|
|
};
|
|
class MemInit {
|
|
C m = s;
|
|
};
|
|
|
|
namespace std {
|
|
typedef decltype(sizeof(int)) size_t;
|
|
|
|
// libc++'s implementation
|
|
template <class _E> class initializer_list {
|
|
const _E *__begin_;
|
|
size_t __size_;
|
|
|
|
initializer_list(const _E *__b, size_t __s) : __begin_(__b), __size_(__s) {}
|
|
|
|
public:
|
|
typedef _E value_type;
|
|
typedef const _E &reference;
|
|
typedef const _E &const_reference;
|
|
typedef size_t size_type;
|
|
|
|
typedef const _E *iterator;
|
|
typedef const _E *const_iterator;
|
|
|
|
initializer_list() : __begin_(nullptr), __size_(0) {}
|
|
|
|
size_t size() const { return __size_; }
|
|
const _E *begin() const { return __begin_; }
|
|
const _E *end() const { return __begin_ + __size_; }
|
|
};
|
|
} // namespace std
|
|
|
|
#if __cplusplus >= 201703L
|
|
|
|
// Test CXXDefaultInitExpr rebuild issue in
|
|
// https://github.com/llvm/llvm-project/pull/87933
|
|
namespace test_rebuild {
|
|
template <typename T, int> class C {
|
|
public:
|
|
C(std::initializer_list<T>);
|
|
};
|
|
|
|
template <typename T> using Ptr = __remove_pointer(T) *;
|
|
template <typename T> C(T) -> C<Ptr<T>, sizeof(T)>;
|
|
|
|
class A {
|
|
public:
|
|
template <typename T1, typename T2> T1 *some_func(T2 &&);
|
|
};
|
|
|
|
struct B : A {
|
|
int *ar = some_func<int>(C{some_func<int>(0)});
|
|
B() {}
|
|
};
|
|
|
|
int TestBody_got;
|
|
template <int> class Vector {
|
|
public:
|
|
Vector(std::initializer_list<int>);
|
|
};
|
|
template <typename... Ts> Vector(Ts...) -> Vector<sizeof...(Ts)>;
|
|
class ProgramBuilder {
|
|
public:
|
|
template <typename T, typename ARGS> int *create(ARGS);
|
|
};
|
|
|
|
struct TypeTest : ProgramBuilder {
|
|
int *str_f16 = create<int>(Vector{0});
|
|
TypeTest() {}
|
|
};
|
|
class TypeTest_Element_Test : TypeTest {
|
|
void TestBody();
|
|
};
|
|
void TypeTest_Element_Test::TestBody() {
|
|
int *expect = str_f16;
|
|
&TestBody_got != expect; // expected-warning {{inequality comparison result unused}}
|
|
}
|
|
} // namespace test_rebuild
|
|
|
|
// Test CXXDefaultInitExpr rebuild issue in
|
|
// https://github.com/llvm/llvm-project/pull/92527
|
|
namespace test_rebuild2 {
|
|
struct F {
|
|
int g;
|
|
};
|
|
struct H {};
|
|
struct I {
|
|
I(const F &);
|
|
I(H);
|
|
};
|
|
struct L {
|
|
I i = I({.g = 0});
|
|
};
|
|
struct N : L {};
|
|
|
|
void f() {
|
|
delete new L; // Ok
|
|
delete new N; // Ok
|
|
}
|
|
} // namespace test_rebuild2
|
|
#endif // __cplusplus >= 201703L
|
|
|
|
#if __cplusplus >= 202002L
|
|
// This test ensures cleanup expressions are correctly produced
|
|
// in the presence of default member initializers.
|
|
namespace PR136554 {
|
|
struct string {
|
|
constexpr string(const char*) {};
|
|
constexpr ~string();
|
|
};
|
|
struct S;
|
|
struct optional {
|
|
template <typename U = S>
|
|
constexpr optional(U &&) {}
|
|
};
|
|
struct S {
|
|
string a;
|
|
optional b;
|
|
int defaulted = 0;
|
|
} test {
|
|
"", {
|
|
{ "", 0 }
|
|
}
|
|
};
|
|
|
|
// Ensure that the this pointer is
|
|
// transformed without crashing
|
|
consteval int immediate() { return 0;}
|
|
struct StructWithThisInInitializer {
|
|
int member() const {
|
|
return 0;
|
|
}
|
|
int m = member() + immediate();
|
|
int m2 = this->member() + immediate();
|
|
};
|
|
|
|
template <typename T>
|
|
struct StructWithThisInInitializerTPL {
|
|
template <typename U>
|
|
int member() const {
|
|
return 0;
|
|
}
|
|
int m = member<int>() + immediate();
|
|
int m2 = this->member<int>() + immediate();
|
|
};
|
|
|
|
void test_this() {
|
|
(void)StructWithThisInInitializer{};
|
|
(void)StructWithThisInInitializerTPL<int>{};
|
|
}
|
|
|
|
struct ReferenceToNestedMembers {
|
|
int m;
|
|
int a = ((void)immediate(), m); // ensure g is found in the correct scope
|
|
int b = ((void)immediate(), this->m); // ensure g is found in the correct scope
|
|
};
|
|
struct ReferenceToNestedMembersTest {
|
|
void* m = nullptr;
|
|
ReferenceToNestedMembers j{0};
|
|
} test_reference_to_nested_members;
|
|
|
|
}
|
|
|
|
|
|
namespace odr_in_unevaluated_context {
|
|
template <typename e, bool = __is_constructible(e)> struct f {
|
|
using type = bool;
|
|
};
|
|
|
|
template <class k, f<k>::type = false> int l;
|
|
int m;
|
|
struct p {
|
|
// This used to crash because m is first marked odr used
|
|
// during parsing, but subsequently used in an unevaluated context
|
|
// without being transformed.
|
|
int o = m;
|
|
p() {}
|
|
};
|
|
|
|
int i = l<p>;
|
|
}
|
|
|
|
#endif
|