This code rejected by clang because t.n
is not compile-time constant.
template<int N>
struct s{
constexpr static int n = N;
};
template<typename T>
void test(T& t){
static_assert(t.n == 1);
}
int main(){
s<1> str;
test(str);
}
But g++ let this go.
Which is the standard-compliant behavior?
One more curious fact is, if I change test
's argument from T& t
to T t
, this will build on both.
What is changed about const-ness?
(I afraid the title may not describe this question correctly, or in detail. Feel free to give me more suitable title)
I believe (but with low confidence) that Clang is correct, because according to the standard, a core constant expression may not contain:
and
t
is a variable of reference type. Assuming (as I do, with low confidence) that "preceding initialization" means initialization that is lexically in scope at the point where the would-be constant expression occurs, since the initialization of a parameter occurs in the scope of the caller,t
does not satisfy this requirement, sot.n
indeed cannot be used in a constant expression.If the parameter type is changed to
T
then this disqualifying bullet point no longer applies.(Even if you accept my interpretation, there is still ambiguity; see CWG 2186)