Wednesday, May 21, 2014

Prevent exceptions from leaving destructors. Now!

Any seasoned C++ programmer should now that permitting an exception to leave the destructor is bad practice, googling for "throw exception destructor" it leads to enough results convincing yourself that is a bad practice (see for example Meyers's "More effective C++" Item 11). Most of the arguments are: "if an object is destroyed during a stack unwinding then throwing an exception it triggers the terminate function" or "if an STL container is being destroyed it start to destroy all his contained elements and given the fact the STL containers do not expect an exception being thrown then it will not complete the destruction of the remaining objects".

If you still are not convinced by those arguments then I hope you will buy at least the following. Let's look at a possible implementation of an unique ptr (apart the -> and * operators):

template <class T>
class AutoPtr {
public:
AutoPtr(T* aPointer)
: thePointer(aPointer)
{}
~AutoPtr() {
delete thePointer;
}
void reset() {
delete thePointer;
thePointer = nullptr;
}
private:
T* thePointer;
};
view raw AutoPtr hosted with ❤ by GitHub
and a possible use:

int main() {
AutoPtr<Bomb> a(new Bomb());
a.reset();
}
view raw AutoPtrMain hosted with ❤ by GitHub
As you can see the AutoPtr::reset() deletes the stored pointer and then is not able to nullify it due the throw, as soon as the "a" instance goes out of scope due the stack unwinding then ~AutoPtr deletes again thePointer. A possible implementation of reset can be the following:

void reset() {
T* tmp = thePointer;
thePointer = nullptr;
delete tmp;
}
view raw AnotherReset hosted with ❤ by GitHub
but unfortunately it not saves you! Indeed in c++11 specification you can "find" the following:
12.4.3: A declaration of a destructor that does not have an exception-specification is implicitly considered to have the same exception-specification as an implicit declaration (15.4).
and again:
Whenever an exception is thrown and the search for a handler (15.3) encounters the outermost block of a function with an exception-specification that does not allow the exception, then, — if the exception-specification is a dynamic-exception-specification, the function std::unexpected() is called (15.5.2), — otherwise, the function std::terminate() is called (15.5.1).
that means that throwing an exception from a DTOR terminates your program and it doesn't matter if a stack unwinding is going on or not.

This simple example

#include <stdexcept>
class Bomb {
public:
~Bomb() { throw std::runtime_error("BOOM"); }
};
int main()
try {
Bomb b;
}
catch(...) {}
view raw SimpleMain hosted with ❤ by GitHub
does generate a crash if compiled in c++11 mode with gcc (4.8 and 4.9) and clang (3.5) while with intel icc 14.01 doens't call the std::unexpected either the std::terminate (time to fill an icc bug?)