Skip to content

Commit 9ac636e

Browse files
committed
[clang] Reject constexpr-unknown values as constant expressions more consistently.
Perform the check for constexpr-unknown values in the same place we perform checks for other values which don't count as constant expressions. While I'm here, also fix a rejects-valid with a reference that doesn't have an initializer. This diagnostic was also covering up some of the bugs here. The existing behavior with -fexperimental-new-constant-interpreter seems to be correct, but the diagnostics are slightly different; it would be helpful if someone could check on that as a followup. Followup to llvm#128409. Fixes llvm#129844. Fixes llvm#129845.
1 parent 9f0a912 commit 9ac636e

File tree

4 files changed

+50
-16
lines changed

4 files changed

+50
-16
lines changed

clang/lib/AST/ExprConstant.cpp

+12-13
Original file line numberDiff line numberDiff line change
@@ -2419,6 +2419,16 @@ static bool CheckLValueConstantExpression(EvalInfo &Info, SourceLocation Loc,
24192419
LVal.getLValueCallIndex() == 0) &&
24202420
"have call index for global lvalue");
24212421

2422+
if (LVal.allowConstexprUnknown()) {
2423+
if (BaseVD) {
2424+
Info.FFDiag(Loc, diag::note_constexpr_var_init_non_constant, 1) << BaseVD;
2425+
NoteLValueLocation(Info, Base);
2426+
} else {
2427+
Info.FFDiag(Loc);
2428+
}
2429+
return false;
2430+
}
2431+
24222432
if (Base.is<DynamicAllocLValue>()) {
24232433
Info.FFDiag(Loc, diag::note_constexpr_dynamic_alloc)
24242434
<< IsReferenceType << !Designator.Entries.empty();
@@ -3597,7 +3607,8 @@ static bool evaluateVarDeclInit(EvalInfo &Info, const Expr *E,
35973607
// expressions here; doing so would regress diagnostics for things like
35983608
// reading from a volatile constexpr variable.
35993609
if ((Info.getLangOpts().CPlusPlus && !VD->hasConstantInitialization() &&
3600-
VD->mightBeUsableInConstantExpressions(Info.Ctx)) ||
3610+
VD->mightBeUsableInConstantExpressions(Info.Ctx) &&
3611+
!AllowConstexprUnknown) ||
36013612
((Info.getLangOpts().CPlusPlus || Info.getLangOpts().OpenCL) &&
36023613
!Info.getLangOpts().CPlusPlus11 && !VD->hasICEInitializer(Info.Ctx))) {
36033614
if (Init) {
@@ -16993,18 +17004,6 @@ bool Expr::EvaluateAsInitializer(APValue &Value, const ASTContext &Ctx,
1699317004

1699417005
if (!Info.discardCleanups())
1699517006
llvm_unreachable("Unhandled cleanup; missing full expression marker?");
16996-
16997-
if (Value.allowConstexprUnknown()) {
16998-
assert(Value.isLValue() && "Expected an lvalue");
16999-
auto Base = Value.getLValueBase();
17000-
const auto *NewVD = Base.dyn_cast<const ValueDecl *>();
17001-
if (!NewVD)
17002-
NewVD = VD;
17003-
Info.FFDiag(getExprLoc(), diag::note_constexpr_var_init_non_constant, 1)
17004-
<< NewVD;
17005-
NoteLValueLocation(Info, Base);
17006-
return false;
17007-
}
1700817007
}
1700917008

1701017009
return CheckConstantExpression(Info, DeclLoc, DeclTy, Value,

clang/test/CodeGenCXX/cxx23-p2280r4.cpp

+12
Original file line numberDiff line numberDiff line change
@@ -13,3 +13,15 @@ int& test() {
1313
auto &i = s;
1414
return i;
1515
}
16+
17+
// CHECK: @_Z1fv(
18+
// CHECK-NEXT: entry:
19+
// CHECK-NEXT: getelementptr inbounds nuw %struct.B
20+
// CHECK-NEXT: getelementptr inbounds nuw %struct.A
21+
// CHECK-NEXT: [[X:%.*]] = load ptr, ptr @x, align 8
22+
// CHECK-NEXT: store ptr [[X]]
23+
int &ff();
24+
int &x = ff();
25+
struct A { int& x; };
26+
struct B { A x[20]; };
27+
B f() { return {x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x}; }

clang/test/SemaCXX/constant-expression-cxx11.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -1472,8 +1472,8 @@ namespace ConvertedConstantExpr {
14721472
enum class E {
14731473
em = m,
14741474
en = n, // expected-error {{enumerator value is not a constant expression}} cxx11_20-note {{initializer of 'n' is unknown}}
1475-
eo = (m + // pre-cxx23-error {{not a constant expression}}
1476-
n // cxx11_20-note {{initializer of 'n' is unknown}} cxx23-error {{not a constant expression}}
1475+
eo = (m + // expected-error {{not a constant expression}}
1476+
n // cxx11_20-note {{initializer of 'n' is unknown}}
14771477
),
14781478
eq = reinterpret_cast<long>((int*)0) // expected-error {{not a constant expression}} expected-note {{reinterpret_cast}}
14791479
};

clang/test/SemaCXX/constant-expression-p2280r4.cpp

+24-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// RUN: %clang_cc1 -std=c++23 -verify %s
1+
// RUN: %clang_cc1 -std=c++23 -verify=expected,nointerpreter %s
22
// RUN: %clang_cc1 -std=c++23 -verify %s -fexperimental-new-constant-interpreter
33

44
using size_t = decltype(sizeof(0));
@@ -154,3 +154,26 @@ int g() {
154154
static_assert(f(arr) == 5);
155155
}
156156
}
157+
158+
namespace GH128409 {
159+
int &ff();
160+
int &x = ff(); // nointerpreter-note {{declared here}}
161+
constinit int &z = x; // expected-error {{variable does not have a constant initializer}}
162+
// expected-note@-1 {{required by 'constinit' specifier here}}
163+
// nointerpreter-note@-2 {{initializer of 'x' is not a constant expression}}
164+
}
165+
166+
namespace GH129845 {
167+
int &ff();
168+
int &x = ff(); // nointerpreter-note {{declared here}}
169+
struct A { int& x; };
170+
constexpr A g = {x}; // expected-error {{constexpr variable 'g' must be initialized by a constant expression}}
171+
// nointerpreter-note@-1 {{initializer of 'x' is not a constant expression}}
172+
const A* gg = &g;
173+
}
174+
175+
namespace extern_reference_used_as_unknown {
176+
extern int &x;
177+
int y;
178+
constinit int& g = (x,y); // expected-warning {{left operand of comma operator has no effect}}
179+
}

0 commit comments

Comments
 (0)