This code fails to compile in most compilers but at first I intuitively expected SFINAE to protect me:
typedef void (*A)();
template < typename T >
st
In C++03 specification, the rule of SFINAE is a little vague, allowing compiler authors to go to any length to find substitution failure to result in SFINAE. The relevant text §14.8.2/2 from C++03 says,
-
[...] If a substitution in a template parameter or in the function type of the function template results in an invalid type, type deduction fails [...]
It further explains few reasons for failure, but none of them actually says at what point the substitution failure should be considered as SFINAE. So I guess, your code may work fine in C++03 (or may not, depending on how the compiler authors interpret the text. It is confusing to me anyway).
But the wordings in C++11 has been improved removing the vagueness. It says in §14.8.2/8,
If a substitution results in an invalid type or expression, type deduction fails. An invalid type or expression is one that would be ill-formed if written using the substituted arguments. [ Note: Access checking is done as part of the substitution process. —end note ] Only invalid types and expressions in the immediate context of the function type and its template parameter types can result in a deduction failure.
The term "immediate context" is interesting, and I think it applies to your situation. More specifically, the substitution failure in the meta-function a_metafun
is not considered "immediate context" of the function-type. It is ill-formed in C++11, not SFINAE.
But then, even though C++11 has introduced the phrase "immediate context" to make the text slightly better, still the definition of the phrase isn't clear enough. Here is one active issue: