std::terminate
problemSection: 17.9.5 [exception.terminate] Status: Open Submitter: Daniel Krügler Opened: 2011-09-25 Last modified: 2016-01-28
Priority: 3
View all issues with Open status.
Discussion:
Andrzej Krzemienski reported the following on comp.std.c++:
In N3290, which is to become the official standard, in 17.9.5.4 [terminate], paragraph 1 reads
Remarks: Called by the implementation when exception handling must be abandoned for any of several reasons (15.5.1), in effect immediately after evaluating the throw-expression (18.8.3.1). May also be called directly by the program.
It is not clear what is "in effect". It was clear in previous drafts where paragraphs 1 and 2 read:
Called by the implementation when exception handling must be abandoned for any of several reasons (15.5.1). May also be called directly by the program.
Effects: Calls theterminate_handler
function in effect immediately after evaluating the throw-expression (18.8.3.1), if called by the implementation, or calls the currentterminate_handler
function, if called by the program.It was changed by N3189. The same applies to function unexpected (D. 11.4, paragraph 1).
Assuming the previous wording is still intended, the wording can be read "unlessstd::terminate
is called by the program, we will use the handler that was in effect immediately after evaluating the throw-expression". This assumes that there is some throw-expression connected to every situation that triggers the call tostd::terminate
. But this is not the case:
- In case
std::thread
is assigned to or destroyed while being joinable there is no throw-expression involved.- In case
std::unexpected
is called by the program,std::terminate
is triggered by the implementation - no throw-expression involved.- In case a destructor throws during stack unwinding we have two throw-expressions involved.
Which one is referred to?
In casestd::nested_exception::rethrow_nested
is called for an object that has captured no exception, there is no throw-expression involved directly (and may no throw be involved even indirectly). Next, 17.9.5.1 [terminate.handler], paragraph 2 saysRequired behavior: A
terminate_handler
shall terminate execution of the program without returning to the caller.This seems to allow that the function may exit by throwing an exception (because word "return" implies a normal return).
One could argue that words "terminate execution of the program" are sufficient, but then why "without returning to the caller" would be mentioned. In case such handler throws, noexcept specification in functionstd::terminate
is violated, andstd::terminate
would be called recursively - shouldstd::abort
not be called in case of recursivestd::terminate
call? On the other hand some controlled recursion could be useful, like in the following technique.
The here mentioned wording changes by N3189 in regard to 17.9.5.4 [terminate] p1
were done for a better separation of effects (Effects element) and additional normative
wording explanations (Remarks element), there was no meaning change intended. Further,
there was already a defect existing in the previous wording, which was not updated when
further situations where defined, when std::terminate
where supposed to be
called by the implementation.
terminate_handler
function, so should be moved just after
"Effects: Calls the current terminate_handler
function."
It seems ok to allow a termination handler to exit via an exception, but the
suggested idiom should better be replaced by a more simpler one based on
evaluating the current exception pointer in the terminate handler, e.g.
void our_terminate (void) { std::exception_ptr p = std::current_exception(); if (p) { ... // OK to rethrow and to determine it's nature } else { ... // Do something else } }
[2011-12-09: Daniel comments]
A related issue is 2111.
[2012, Kona]
Move to Open.
There is an interaction with Core issues in this area that Jens is already supplying wording for. Review this issue again once Jens wording is available.
Alisdair to review clause 15.5 (per Jens suggestion) and recommend any changes, then integrate Jens wording into this issue.
Proposed resolution: