Re: Type traits and accessibility
On 19 Apr., 20:59, Nikolay Ivchenkov <ts...@mail.ru> wrote:
On 19 Apr, 08:46, Daniel Kru"gler <daniel.krueg...@googlemail.com>
wrote:
On 18 Apr., 00:03, Nikolay Ivchenkov <ts...@mail.ru> wrote:
Note also that well-formed expression could render the program ill-
formed. In particular, the program is ill-formed if an undefined
function is used:
#include <type_traits>
struct Y
{
Y(int); // has no definition
};
int main()
{
// OK
!std::is_constructible<Y, int>::value ?
(void)Y(0) : (void)sizeof Y(0);
// renders the program ill-formed
std::is_constructible<Y, int>::value ?
(void)Y(0) : (void)sizeof Y(0);
}
[I'm referring to your revised version in the
following]
Not really. Your test program which contains
use<!std::is_constructible<Y, int>::value>();
is ill-formed, because the definition of the
constructor is *missing*, but it is *not* ill-
formed, because an undefined entity is *used*.
Both conditions are important - unless the following wording applies:
"If no valid specialization can be generated for a template
definition, and that template is not instantiated, the template
definition is ill-formed, no diagnostic required." - in this case I'm
not sure, but I can propose for consideration the following:
#include <type_traits>
struct Y
{
Y(int); // has no definition
Y(long) {}
};
template <int n>
void use()
{
// renders the program ill-formed when n==1
static_cast<Y>(
typename std::conditional<n==1, int, long>::type());
}
template <>
void use<0>()
{
// OK
sizeof static_cast<Y>(0);
}
int main()
{
// calls use<false>
use<!std::is_constructible<Y, int>::value>();
// calls use<true>
use<std::is_constructible<Y, int>::value>();
}
We can conclude that this program is ill-formed considering two facts
(but not only one of them):
1) the constructor Y(int) is used (so, it shall be defined), and
2) the definition of the constructor Y(int) is not provided.
These facts collectively can be expressed as "undefined function is
used".
This example doesn't matter in this discussion, because it does
not properly reflect the definition of std::is_constructible.
If you look at the definition of std::is_constructible, it obviously
does not describe a complete program, so we cannot argue
about possibly missing definitions of any potentially used
entity of this definition, because that cannot be deduced from
the definition. It was also intended *not* to use std::declval
instead of the seemingly near-to-equal create function template,
because we would have to consider the rules of ill-formed
code related to potentially evaluated call of std::declval (which
makes the code ill-formed). From the given spec we can also
not conclude that the definition of the used specialization
of the function template create is missing which otherwise
would be a reason to make a completed program ill-formed.
In other words: Any *additional* programming errors contained
in a program using std::is_constructible (like any missing
definition of a used entity) does not affect the definition of the
outcome of this type-trait - These kinds of violations can still
be ill-formed with an unspecified location/point in time, when
this will be diagnosed, following the basic rules described in
sub-clause 1.4 [intro.compliance].
Similarly, the following program
#include <typeinfo>
class X;
int main()
{
typeid(X);
}
is ill-formed because:
1) the class type X is used as the operand of the typeid operator
(i.e. in the context where it is required to be completely-defined),
and
2) the definition of the class X is not provided prior to the typeid
expression.
Is the expression typeid(X) well-formed?
It is clearly ill-formed, but that does not affect in any case
the situation with is_constructible and is_convertible. It could
potentially be, if typeid would be involved within the definition
of either of these compiler support traits.
So, it is very important to distinguish "ill-formed program" and "ill-
formed expression".
Not really. The core language does not distinguish
between a well/ill-formed program or a well/ill-
formed expression. If the core language speaks
if an ill-formed expression this is just a short
wording of "A program that contains this expression
is ill-formed".
I am talking about the inverse consequence: we can't strictly conclude
that an expession is ill-formed when the specification asserts ill-
formedness of the entire program only.
I don't see how this affects the definition of is_constructible,
as shown above. The definition itself only shows a partial
program, so users can still make an ill-formed program
and cannot expect that the pure existence of std::is_constructible
make the code well-formed and would only affect the outcome
of the evaluation of std::is_constructible</types/>::value.
There is no problem with all of these examples, they
are covered by the definition of an ill-formed
expression equivalent being to an ill-formed program.
Where is this definition? For the program
#include <type_traits>
struct B {};
struct B1 : B {};
struct B2 : B {};
struct D : B1, B2 {};
template <class T>
typename std::add_rvalue_reference<T>::type create();
typedef char one[1];
typedef char two[2];
template <class T>
one &f(char (*)[sizeof static_cast<T>(create<D *>())]);
template <class T>
two &f(...);
int main()
{
static_assert(sizeof f<int>(0) == sizeof(two), "");
static_assert(sizeof f<B *>(0) == sizeof(two), "");
}
GNU C++ 4.5 issues the following diagnostic message:
"error: 'B' is an ambiguous base of 'D'".
Presumably, the expression
sizeof static_cast<int>(create<D *>())
is treated as ill-formed, while the expression
sizeof static_cast<B *>(create<D *>())
is treated as well-formed (but the entire program is considered as ill-
formed). If you think that the behavior of the GNU C++ compiler is
wrong, how can you prove it?
This is a compiler defect, because it violates 14.9.2 [temp.deduct].
Unless the compiler vendor argues that this behaviour is intended
because of his/her reading of the core language this should be
considered as a compiler bug. I have at least weak evidence that
this my interpretation is agreed on as a compiler defect based on
an *informal* query to Jason Merrill. I suggest to open a gcc bug
entry for this and let's see to what kind of further insight this may
lead.
HTH & Greetings from Bremen,
Daniel Kr?gler
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]