Re: Using noexcept
Am 17.06.2011 09:48, schrieb Rani Sharoni:
the compiler would still have to generate code to test
whether an exception is trying to leave the function,
and then call std::terminate,
True. the problematic case is when the throw can be potentially masked
by outer catch.
For example:
try {
function_with_noexcept();
might_throw();
extern_c_function();
}
catch(...) {}
extern-c function (e.g. OS call) might not be aware about the noexcept
state-flag hence it's the caller's responsibility to toggle the flag
before/after calling into every throw/nothrow (might skip some).
I afraid I couldn't quite follow you, could you please elaborate?
I personally wanted UB to push such bug handling under QOI but it
seems that the committee though that UB for such unhandled exceptions
is somewhat bad PR for a language.
This sounds like a strange argument to me. Actually, there is lots of UB
already in C++ as it is right now, and it has ever been this way - if
the user does something stupid, well,...
OTOH, mandating mitigation for such
specific bug seems inconsistent with other cases that traditionally
are left for UB (e.g. access bogus memory).
Indeed.
I also had no problem with breaking the legacy throw() requirement on
violation since the popular MSVC compiler is doing that since always
(hence favoring optimizations).
Indeed.
whereas if I had *no* exception specification,
no such code would have to be generated in first place *at all*.
IOW, you get less optimal code by adding "noexcept" -
This is not true. The compiler can avoid generating unwind code for
the throwing case which is the main no-throw optimization (caller and
callee). for example:
unique_ptr<A> p = new A;
nothrow_call(p);
// implicitly generate delete p, no need for special unwind on-throw
code
return;
Ok, I see what you mean, but I don't quite agree with the argument.
Allow me to elaborate: Without a throw specification, the compiler has
to generate the unwind code, but this is usually "for free" since it is
not the common call path. Gcc generates rather a table in the binary
which code to call if stack unwinding is required. If I add a throw
specification to a calling function, the compiler has to generate both,
the stack unwinding code, and extra code to check whether the exception
thrown was actually specified, and if not, run into std::unexpected (or
something like this, I forgot the nasty details). Whereas now, with
noexcept, I get only half of the optimization. The compiler has *still*
to generate code to check whether an exception wants to leave the
function. It may only omit the unwind code. But that seems to be the
easy part for me (though maybe I'm wrong) the hard part is to check
whether there is actually any exception on the way.
Whereas if it would have been UB to throw an exception in noexcept, the
compiler would have to do nothing *at all* to handle such a case, and
thus there would be no code at all - hence no unwind code, and no
exception checking code.
so in which sense is this an improvement?
The nexcept(condition) is important for no-throw based user
optimization (e.g. move if no-throw).
That was the initial motivation as I understand. Well, maybe. Time will
show.
I would also have understood if noexcept would be statically checked
since then I had means to validate at compile time
Static validation is problematic in general due to conditional no-
throw functions (runtime).
For example:
vector<int> v(10); // reserve space for 10 elements
v.push_back(1); // no-throw since no allocation needed
map<int, int> m;
m[0] = 0;
m[0] = 1; // no-throw on replace
It seems that for static checking, noexcept needs to have another
flavor - "unconditionally might throw".
I don't think the compiler could be expected to prove in such cases that
the calls cannot throw. Actually, if I would have the *requirement* to
write a function that cannot throw, for whatever reason, I wouldn't use
the corresponding standard library functions, but rather build something
on my own. That is, you're on yourself if you need additional
requirements like static memory or static code. std::vector::push_back()
cannot be marked as noexcept since it may call new, and hence it isn't
noexcept. If you use it in a noexcept function, well, you have to pay
for the price and add a try-catch block around it. This block wouldn't
do anything except adding additional code - hopefully:
try {
v.push_back(1);
} catch(..) {
// Ooops, how did that happen now?
std::terminate();
}
IOW, it would require from the user to *manually* insert code the
compiler would have to insert otherwise to indicate that "yes, I know
this isn't optimal, I don't get the optimizations I might, but at least
its safe enough".
If you cannot live with that, well, use arrays, they are noexcept for sure.
This is probably much more common than conditional no-throw, For
example:
void* operator new(size_t) noexcept(--);
void foo() noexcept
{
new A; // can be statically rejected
}
I don't understand? void *operator new(size_t) is definitely not
"noexcept" if the allocation fails. Unless you implement your own
version, which terminates the program manually if it runs out of memory,
and thus can be marked as "noexcept". In which case of course the above
code should compile. Of course, it is then up to the programmer
implementing oerator new() to ensure that the noexcept guarantee isn't
violated, but this is again beyond what the compiler can do.
Note that the MSVC compiler issues a warning when explicit 'throw' is
used unguarded inside a throw() function.
Well, I'm not using MSVC but rather gcc, but whatever; while a nice
"service" of the compiler, I would rather prefer to have this an error.
If you violate the contract, the compiler should error, not warn.
The following is exactly the same problem, and AFAIK an error in C++.
int foo(int args)
{
return; // oops!
}
Why isn't that a warning either? Because it breaks the contract. In the
same vain, throwing an exception in noexcept breaks the contract, and
hence should be an error. Hence, everything that *potentially* may throw
cannot be used in noexcept - unless the programmer knows that he has to
a pay a price for this, and add a try-catch around the code. Even *if*
he knows that the catch() will not be invoked due to reasons beyond what
the compiler may test for. It is just a safety feature in case the code
is modified later on though forgetting about the precise conditions
under which it was designed (such situations appear).
The above non-standard idea extends such for other functions beside of
the 'throw' operator.
One concern about having such is with breaking the compilation of lots
of existing code that doesn't care about conditions such as out of
memory.
I don't quite understand? Current code doesn't use noexcept, thus there
shouldn't be a problem?
I think that static checking is nice to have but not super crucial
since anyway proper testing is needed given the many other bugs that
can be detected only at runtime or using custom static analysis tools
(e.g. all the traditional UB bugs).
Again, this sounds like a strange argument to me. I would prefer to
fixup the language - "just because we've been sloppy in the past, let's
continue to be sloppy". (-; No, I don't think so.
So long,
Thomas
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]