While thinking about this question, I stumbled upon something else I don't understand.
Standard says...
If a class has no user-declared destructor, a destructor is implicitly declared as defaulted. An implicitly-declared destructor is an inline public member of its class.
[...] If a class has a base class with a virtual destructor, its destructor (whether user- or implicitly-declared) is virtual.
A destructor that is defaulted and not defined as deleted is implicitly defined when it is odr-used or when it is explicitly defaulted after its first declaration.
[...] A virtual member function is odr-used if it is not pure. [...]
So now I'm wondering whether this code should compile:
#include <memory>
struct Base {
virtual ~Base() = default;
};
struct Bar;
struct Foo : Base {
std::unique_ptr<Bar> bar_{};
};
I would think that ~Foo()
has to be implicitly defined, because it is virtual, but it would not compile because Bar
is incomplete in this TU. Yet the code compiles in all major compilers.
What am I missing?
OK, since the issue has been clarified by various answers and comments that have been posted, let me give another shot at answering it. References are to the C++17 standard.
The issue with OP's code is that
Foo
has a virtual destructor that is not pure, and all non-pure virtual functions are implicitly odr-used ([basic.def.odr]/3). Since the destructor is odr-used, it seems that the implementation must generate a definition ([class.dtor]/7), and that definition must result in the instantiation ofstd::default_delete<Bar>::operator()
, which makes the program ill-formed becauseBar
is incomplete ([unique.ptr.dltr.dflt]/4). So why do compilers not produce a diagnostic?I think there is an issue with the wording of [class.dtor]/7, which reads as follows and has not been substantially changed in newer standard editions to date:
When exactly is the implicit definition of
Foo::~Foo
generated? It is generated "when"Foo::~Foo
is odr-used. But... when isFoo::~Foo
odr-used?Obviously "when" does not refer to physical time, and probably means something like "where". And that presumably implies that:
But in the case of an implicit odr-use of a virtual destructor which occurs purely by dint of it being virtual, where is it odr-used? One way to interpret it is that, in every translation unit where a definition of class
Foo
appears,Foo::~Foo
is considered odr-used in that translation unit, and the compiler must behave as if it generates a definition in that translation unit. If that's the case, then your program requires a diagnostic.Another way to interpret it is that since the standard doesn't define any place where the implicit odr-use occurs, it's considered unspecified, and the implementation can choose where to define the destructor, or even not at all. (That is, if "when" means "where", and "where" means "in the set of places where the odr-use occurs", it's unspecified what that set actually is, and it might be empty.) In practice, compilers probably define the destructor in translation units where it is "odr-used by the vtable" (I put it in quotation marks because technically the standard doesn't define the concept of a vtable, and so a vtable cannot truly odr-use anything) and the vtable is "odr-used" by certain expressions such as potential invocations of the constructor,
dynamic_cast
,typeid
, and exception handling involvingFoo
. And if you don't do any of those things, the compiler doesn't implicitly define the virtual destructor in that TU.I think that the standard wording should probably be amended in order to codify existing practice, i.e., the standard should say that it's unspecified where the implicit odr-use of virtual functions occurs. (Also, if the implementation does decide that the answer is "nowhere", it shouldn't be allowed to then treat the program as ill-formed NDR under [basic.def.odr]/4. That would be perverse.)