I came across an inconsistency in the way current C++ compilers (clang/gcc) determine whether a name is dependent. In the following example, A::f
is dependent but ::f
is not, resulting in an error when the latter is used.
template<typename>
struct B
{
typedef int Type;
};
template<typename U>
static U f(U u);
template<typename T>
struct A
{
template<typename U>
static U f(U u);
typename B<decltype(f(0))>::Type m1; // typename required
B<decltype(::f(0))>::Type m2; // typename not required
};
The inconsistent part is that the declaration of A::f
does not depend on a template parameter of A
, meaning it seems unnecessary to treat it as a dependent name.
This behaviour seems to be covered by the following wording in the C++11 standard:
[temp.dep.expr]/3
An id-expression is type-dependent if it contains
- an identifier associated by name lookup with one or more declarations declared with a dependent type
[temp.dep.type]/3
A type is dependent if it is
- a compound type constructed from any dependent type
The declaration of ::f
is clearly not dependent, as its type depends only on its own template parameters. Why should A::f
be treated differently?
I think that based on the standard,
f
is non-dependent, actually.This applies equally to the global template function as it does to the member template function: not at all. The return type of
U
is dependent inside the definitions of the template functions, but for the caller, the function type off<int>
has already been transformed fromU(U)
toint(int)
. At any rate, it wouldn't explain why compilers treat the two cases differently, and it also would not explain why a non-template member function is also treated as dependent.These do not apply. There is no
<
or>
that must always be present in a template-id, and there is no conversion function being called.See below.
This also does not apply: there are no arrays involved.
So it depends on whether
f
is a member of an unknown specialization. But it isn't:These cannot apply:
f
is neither qualified nor part of a class member access expression.Since the only way
f
can be dependent is if it is a member of an unknown specialization, and it is not a member of an unknown specialization,f
must not be dependent.As for why compilers nonetheless treat it as dependent, I have no answer. Either some part of my answer here is wrong, the compilers have bugs, or the compilers follow a different version of the C++ standard. Testing with an example that works regardless of whether names are dependent shows a few variations in compiler treatment:
This difference in clang's treatment of
b()
,decltype(b())()
, andc<S1>()
is particularly troubling to me. It just doesn't make any sense. They're clearly all equally dependent. I can understand from an implementation point-of-view that care must be taken not to generate code for the member functions just yet because there might be specialisations ofS2<S1>::b
orS2<S1>::c<S1>
, but that applies to all, and has no effect on the return type.