digitalmars.D - Performance of exception handling
- Alexander (18/18) Apr 26 2011 Hi,
- Vladimir Panteleev (7/11) Apr 26 2011 On my Windows box with an i7 920, a simple try/throw/catch loop runs at ...
- Alexander (11/13) Apr 26 2011 I've not found any references to setjmp/longjmp, but what I've found -...
- Vladimir Panteleev (9/10) Apr 26 2011 Sounds spot-on to me :)
- Denis Koroskin (4/24) Apr 26 2011 Full stack trace resolving could be done lazily, but you need to save ca...
- Jacob Carlborg (5/14) Apr 26 2011 SEH is only available on Windows. On all other platforms DWARF
- bearophile (4/6) Apr 26 2011 On Windows I have timed DMD exceptions to be about 12 times slower than ...
- Kagamin (2/4) Apr 26 2011 Did you compare with c++ dwarf implementation?
- Alexander (3/4) Apr 26 2011 Yes, c++ is significantly faster. But it seems that my problem is solv...
- Daniel Gibson (5/12) Apr 26 2011 It'd be nice to have fast(er) exceptions and still be able to get
- Alexander (22/24) Apr 26 2011 Well, quick-and-dirty solution for this (uncaught only, though) follow...
- Sean Kelly (6/20) Apr 26 2011 Right now, traces are generated on throw. It should be possible to gener...
- Alexander (3/4) Apr 26 2011 Why not? My brute-force approach works (temporarily disabling traceHan...
- Sean Kelly (13/18) Apr 26 2011 traceHandler), so why it couldn't be done more nicely somewhere in deh2 ...
- Denis Koroskin (19/24) Apr 26 2011 Interesting enough, it is already done lazily (in toString(), which I
- Sean Kelly (13/21) Apr 26 2011 generate them on catch instead. The performance would be the same either...
- Steven Schveighoffer (23/37) Apr 26 2011 If it's possible to generate them on catch, couldn't you request trace
-
Sean Kelly
(22/50)
Apr 26 2011
wrote: - Denis Koroskin (23/44) Apr 26 2011 You might have misunderstood me, or I might be wrong, but I believe you ...
-
Sean Kelly
(36/66)
Apr 26 2011
wrote: - Andrej Mitrovic (17/17) May 09 2011 I'll have to agree that exceptions are quite slow. I was just testing
- Walter Bright (3/14) May 09 2011 Validating input data should not be done with exceptions. The exceptions...
- Andrej Mitrovic (3/3) May 09 2011 Well I can understand throwing exceptions when using readln() or
- Walter Bright (2/5) May 09 2011 Perhaps decode() is badly designed.
- Andrej Mitrovic (2/7) May 09 2011 It has goto's and exceptions. What's not to like? :p
- Andrej Mitrovic (2/11) May 09 2011 Oh and an enforce at the top to kill any potential inlining. Hehehe.
- Robert Jacques (5/10) May 10 2011 Well, you are supposed to use validate first on any untrusted input. Now...
- Alexander (3/4) May 10 2011 Not to mention that validating first and then running again through the ...
- Jonathan M Davis (10/35) Apr 26 2011 Yeah. Exceptions are _slow_. I ran into some problems with that when rew...
Hi, I'm doing some benchmarks (DMD 2.052 on Linux), and noticed that exception handling is terrible slow even on quite fast hardware. There is a note on DM's site, saying: "Because errors are unusual, execution of error handling code is not performance critical." Well, sure this is (somehow) true, but *so* slow? In my tests, Xeon 3.4 GHz is able to handle only ca. 1000 exceptions/s (!). 1ms for single exception is a little bit too much, especially when application needs to recover fast, not to mention that on slower hardware (like Atom) it will be really slow. And, since some exceptions are not so unusual in normal flow (file not found, connection reset etc), "not performance critical" is not really applicable. Simple example - in case of web (or any other) server with high load, socket exceptions of any kind are quite common. It is still natural to use try/catch to handle them, but it will slow down everything else (imagine > 1K connections/s and 10% cut rate). So, my question - is there something, that can improve performance? Any clues where to dig for this? Thank you! Best regards, /Alexander
Apr 26 2011
On Tue, 26 Apr 2011 13:08:33 +0300, Alexander <aldem+dmars nk7.net> wrote:Hi, I'm doing some benchmarks (DMD 2.052 on Linux), and noticed that exception handling is terrible slow even on quite fast hardware.On my Windows box with an i7 920, a simple try/throw/catch loop runs at about 130000 iterations per second. Perhaps DMD doesn't use SEH on Linux, and instead uses setjmp/longjmp? -- Best regards, Vladimir mailto:vladimir thecybershadow.net
Apr 26 2011
On 26.04.2011 12:57, Vladimir Panteleev wrote:On my Windows box with an i7 920, a simple try/throw/catch loop runs at about 130000 iterations per second.Well, g++ with same loop on same linux system gives ca. 160000 iter/s, which is quite OK for me.Perhaps DMD doesn't use SEH on Linux, and instead uses setjmp/longjmp?I've not found any references to setjmp/longjmp, but what I've found - disabling trace handler with "Runtime.traceHandler = null" boost performance significantly - in my case I got 1600000 iter/s (wow!), which is perfectly OK. AFAIK, traceHandler is something that prints out the stack trace, providing valuable information only when there is no catch. If there is a catch, then, obviously, this slows down exception processing significantly without any need (it is called on every invocation of throw). I am right? /Alexander
Apr 26 2011
On Tue, 26 Apr 2011 15:43:26 +0300, Alexander <aldem+dmars nk7.net> wrote:I am right?Sounds spot-on to me :) I recall there was a similar performance problem due to stack traces, with the "priority send" function in the message queue implementation. The "priority send" function constructed an exception object, which made it slower than the normal "send" function. -- Best regards, Vladimir mailto:vladimir thecybershadow.net
Apr 26 2011
On Tue, 26 Apr 2011 16:43:26 +0400, Alexander <aldem+dmars nk7.net> wrote:On 26.04.2011 12:57, Vladimir Panteleev wrote:Full stack trace resolving could be done lazily, but you need to save call stack addresses at throw side (which is very quick anyway). Could you please file a bugzilla enhancement request for that?On my Windows box with an i7 920, a simple try/throw/catch loop runs at about 130000 iterations per second.Well, g++ with same loop on same linux system gives ca. 160000 iter/s, which is quite OK for me.Perhaps DMD doesn't use SEH on Linux, and instead uses setjmp/longjmp?I've not found any references to setjmp/longjmp, but what I've found - disabling trace handler with "Runtime.traceHandler = null" boost performance significantly - in my case I got 1600000 iter/s (wow!), which is perfectly OK. AFAIK, traceHandler is something that prints out the stack trace, providing valuable information only when there is no catch. If there is a catch, then, obviously, this slows down exception processing significantly without any need (it is called on every invocation of throw). I am right? /Alexander
Apr 26 2011
On 2011-04-26 12:57, Vladimir Panteleev wrote:On Tue, 26 Apr 2011 13:08:33 +0300, Alexander <aldem+dmars nk7.net> wrote:SEH is only available on Windows. On all other platforms DWARF exceptions (I think that's what they're called) are used. -- /Jacob CarlborgHi, I'm doing some benchmarks (DMD 2.052 on Linux), and noticed that exception handling is terrible slow even on quite fast hardware.On my Windows box with an i7 920, a simple try/throw/catch loop runs at about 130000 iterations per second. Perhaps DMD doesn't use SEH on Linux, and instead uses setjmp/longjmp?
Apr 26 2011
Alexander:I'm doing some benchmarks (DMD 2.052 on Linux), and noticed that exception handling is terrible slow even on quite fast hardware.On Windows I have timed DMD exceptions to be about 12 times slower than (irreducible) Java ones. I have done only a limited amount of different benchmarks on this. Bye, bearophile
Apr 26 2011
Alexander Wrote:I'm doing some benchmarks (DMD 2.052 on Linux), and noticed that exception handling is terrible slow even on quite fast hardware.Did you compare with c++ dwarf implementation?
Apr 26 2011
On 26.04.2011 15:13, Kagamin wrote:Did you compare with c++ dwarf implementation?Yes, c++ is significantly faster. But it seems that my problem is solved already. /Alexander
Apr 26 2011
Am 26.04.2011 15:16, schrieb Alexander:On 26.04.2011 15:13, Kagamin wrote:It'd be nice to have fast(er) exceptions and still be able to get backtraces for uncaught exceptions (or even for caught ones on demand). Cheers, - DanielDid you compare with c++ dwarf implementation?Yes, c++ is significantly faster. But it seems that my problem is solved already. /Alexander
Apr 26 2011
On 26.04.2011 15:20, Daniel Gibson wrote:It'd be nice to have fast(er) exceptions and still be able to get backtraces for uncaught exceptions (or even for caught ones on demand).Well, quick-and-dirty solution for this (uncaught only, though) follows. Not very nice, but "works for me" :) ---snip--- import core.runtime; int main(string[] argv) { auto oldTraceHandler = Runtime.traceHandler; Runtime.traceHandler = null; try { return try_main(argv); } catch (Throwable ex) { Runtime.traceHandler = oldTraceHandler; throw ex; } } int try_main(string[] argv) { // Real main() code } ---snip---
Apr 26 2011
Right now, traces are generated on throw. It should be possible to generate t= hem on catch instead. The performance would be the same either way however. I= t would be nice to generate them lazily but I don't think that's possible.=20= Sent from my iPhone On Apr 26, 2011, at 6:20 AM, Daniel Gibson <metalcaedes gmail.com> wrote:Am 26.04.2011 15:16, schrieb Alexander:already.On 26.04.2011 15:13, Kagamin wrote: =20Did you compare with c++ dwarf implementation?=20 Yes, c++ is significantly faster. But it seems that my problem is solved==20 /Alexander=20 It'd be nice to have fast(er) exceptions and still be able to get backtraces for uncaught exceptions (or even for caught ones on demand). =20 Cheers, - Daniel
Apr 26 2011
On 26.04.2011 18:14, Sean Kelly wrote:It would be nice to generate them lazily but I don't think that's possible.Why not? My brute-force approach works (temporarily disabling traceHandler), so why it couldn't be done more nicely somewhere in deh2 (I couldn't figure out yet, how exactly)? /Alexander
Apr 26 2011
On Apr 26, 2011, at 9:18 AM, Alexander wrote:On 26.04.2011 18:14, Sean Kelly wrote: =20possible.=20It would be nice to generate them lazily but I don't think that's ==20 Why not? My brute-force approach works (temporarily disabling =traceHandler), so why it couldn't be done more nicely somewhere in deh2 = (I couldn't figure out yet, how exactly)? Is your brute-force approach faster for entire app execution, or just = try_main() execution? It looks to me like you're just offloading = tracing to the catch point, which in the case of an unhandled exception = is after main() exits. By lazily, I meant generating the trace in = toString() or similar. It might work for the typical case where the = trace is pulled in the catch block or not at all, but if the exception = is held until later before being inspected the lazy approach won't work. = Thread does this, for example, when it rethrows an uncaught exception = from a call to join().=
Apr 26 2011
On Tue, 26 Apr 2011 20:14:05 +0400, Sean Kelly <sean invisibleduck.org> wrote:Right now, traces are generated on throw. It should be possible to generate them on catch instead. The performance would be the same either way however. It would be nice to generate them lazily but I don't think that's possible. Sent from my iPhoneInteresting enough, it is already done lazily (in toString(), which I believe should also cache result when it's called first time), but it can be improved a bit. Here is what it currently does: class DefaultTraceInfo : Throwable.TraceInfo { this() { static enum MAXFRAMES = 128; void*[MAXFRAMES] callstack; numframes = backtrace( callstack, MAXFRAMES ); // 1 framelist = backtrace_symbols( callstack, numframes ); // 2 } // ... } While the first line is definitely required (otherwise stack information gets lost), I believe the second one is not and can be moved to opApply (or somewhere else).
Apr 26 2011
On Apr 26, 2011, at 9:29 AM, Denis Koroskin wrote:On Tue, 26 Apr 2011 20:14:05 +0400, Sean Kelly =<sean invisibleduck.org> wrote:=20generate them on catch instead. The performance would be the same either = way however. It would be nice to generate them lazily but I don't think = that's possible.Right now, traces are generated on throw. It should be possible to =believe should also cache result when it's called first time), but it = can be improved a bit. The readable version is generated in toString, but the actual trace = occurs on throw. This pretty much requires a memory allocation to store = the trace info, which is one cause for the performance hit. Originally, I didn't have the tracing enabled by default, but no one = seemed to like that. I could try enabling it only for non-release = builds if that would be preferable.==20 Sent from my iPhone =20=20 Interesting enough, it is already done lazily (in toString(), which I =
Apr 26 2011
On Tue, 26 Apr 2011 14:26:00 -0400, Sean Kelly <sean invisibleduck.org> wrote:On Apr 26, 2011, at 9:29 AM, Denis Koroskin wrote:If it's possible to generate them on catch, couldn't you request trace generation while inside the catch block (or at the beginning of a catch block) if you intend to use it? That is: try { ... } catch(Exception e) { e.generateTrace(); // only allowed at the start of the catch block printException(e); // if you haven't generated the trace by now, it's gone. } You'd need compiler help for this probably. But I agree with the OP, 99% of the time you are catching exceptions you do not need the trace.On Tue, 26 Apr 2011 20:14:05 +0400, Sean Kelly <sean invisibleduck.org> wrote:Right now, traces are generated on throw. It should be possible to generate them on catch instead. The performance would be the same either way however. It would be nice to generate them lazily but I don't think that's possible.I can't help but think that using the GC may be overkill here. At the very least, you can use C malloc which is usually much faster. The exception should encapsulate the memory allocation and deallocation completely. -SteveInteresting enough, it is already done lazily (in toString(), which I believe should also cache result when it's called first time), but it can be improved a bit.The readable version is generated in toString, but the actual trace occurs on throw. This pretty much requires a memory allocation to store the trace info, which is one cause for the performance hit.
Apr 26 2011
On Apr 26, 2011, at 11:39 AM, Steven Schveighoffer wrote:On Tue, 26 Apr 2011 14:26:00 -0400, Sean Kelly =<sean invisibleduck.org> wrote:=20<sean invisibleduck.org> wrote:On Apr 26, 2011, at 9:29 AM, Denis Koroskin wrote: =20On Tue, 26 Apr 2011 20:14:05 +0400, Sean Kelly =generate them on catch instead. The performance would be the same either = way however. It would be nice to generate them lazily but I don't think = that's possible.=20Right now, traces are generated on throw. It should be possible to ==20 If it's possible to generate them on catch, couldn't you request trace =generation while inside the catch block (or at the beginning of a catch = block) if you intend to use it?=20 That is: =20 try { ... } catch(Exception e) { e.generateTrace(); // only allowed at the start of the catch block printException(e); // if you haven't generated the trace by now, =it's gone.} =20 You'd need compiler help for this probably. But I agree with the OP, =99% of the time you are catching exceptions you do not need the trace. I suppose it's a matter of how easy this stuff should be to use. I = generally like the default behavior to be the most foolproof.I believe should also cache result when it's called first time), but it = can be improved a bit.Interesting enough, it is already done lazily (in toString(), which =occurs on throw. This pretty much requires a memory allocation to store = the trace info, which is one cause for the performance hit.=20 The readable version is generated in toString, but the actual trace ==20 I can't help but think that using the GC may be overkill here. At the =very least, you can use C malloc which is usually much faster. The = exception should encapsulate the memory allocation and deallocation = completely. The DefaultTraceInfo object is allocated via the GC, and the trace info = itself is allocated via malloc() inside the backtrace() call. See = core.runtime for the details--the code is pretty succinct.=
Apr 26 2011
On Tue, 26 Apr 2011 22:26:00 +0400, Sean Kelly <sean invisibleduck.org> wrote:On Apr 26, 2011, at 9:29 AM, Denis Koroskin wrote:You might have misunderstood me, or I might be wrong, but I believe you only need to store backtrace at the point where exception occurs, and resolve the symbols later (e.g. at toString()). Both are currently done in the DefaultTraceInfo ctor (see the code that you snipped). Collecting the backtrace is very fast, and doesn't allocate on its own: static enum MAXFRAMES = 128; void*[MAXFRAMES] callstack; numframes = backtrace( callstack, MAXFRAMES ); It's the "backtrace_symbols" that does all the hard work, I believe, and as such moving it out of the DefaultTraceInfo ctor should be enough. On the other had, DefaultTraceInfo could be made a struct and be moved to the Exception class to minimize allocations if you feel that exceptions should allocate as little as possible (if any at all, because they may end up throwing OutOfMemory exception). Side question: we don't create OutOfMemory instances, throwing predefined ones instead, right? I don't think that work with exception chaining, does it? I mean, what if we e.g. catch an OOM, try to add a log entry, try to allocate and end up with OOM again. An exception chaining implemented with an intrusive linked list (i.e. implemented with an embedded Throwable next;) will not work for this case. Does druntime handle this case correctly?On Tue, 26 Apr 2011 20:14:05 +0400, Sean Kelly <sean invisibleduck.org> wrote:The readable version is generated in toString, but the actual trace occurs on throw. This pretty much requires a memory allocation to store the trace info, which is one cause for the performance hit. Originally, I didn't have the tracing enabled by default, but no one seemed to like that. I could try enabling it only for non-release builds if that would be preferable.Right now, traces are generated on throw. It should be possible to generate them on catch instead. The performance would be the same either way however. It would be nice to generate them lazily but I don't think that's possible. Sent from my iPhoneInteresting enough, it is already done lazily (in toString(), which I believe should also cache result when it's called first time), but it can be improved a bit.
Apr 26 2011
On Apr 26, 2011, at 11:56 AM, Denis Koroskin wrote:On Tue, 26 Apr 2011 22:26:00 +0400, Sean Kelly =<sean invisibleduck.org> wrote:=20<sean invisibleduck.org> wrote:On Apr 26, 2011, at 9:29 AM, Denis Koroskin wrote: =20On Tue, 26 Apr 2011 20:14:05 +0400, Sean Kelly =generate them on catch instead. The performance would be the same either = way however. It would be nice to generate them lazily but I don't think = that's possible.=20Right now, traces are generated on throw. It should be possible to =I believe should also cache result when it's called first time), but it = can be improved a bit.=20 Sent from my iPhone =20=20 Interesting enough, it is already done lazily (in toString(), which =occurs on throw. This pretty much requires a memory allocation to store = the trace info, which is one cause for the performance hit.=20 The readable version is generated in toString, but the actual trace =seemed to like that. I could try enabling it only for non-release = builds if that would be preferable.=20 Originally, I didn't have the tracing enabled by default, but no one ==20 You might have misunderstood me, or I might be wrong, but I believe =you only need to store backtrace at the point where exception occurs, = and resolve the symbols later (e.g. at toString()). Both are currently = done in the DefaultTraceInfo ctor (see the code that you snipped).=20 Collecting the backtrace is very fast, and doesn't allocate on its =own:=20 static enum MAXFRAMES =3D 128; void*[MAXFRAMES] callstack; =20 numframes =3D backtrace( callstack, MAXFRAMES ); =20 It's the "backtrace_symbols" that does all the hard work, I believe, =and as such moving it out of the DefaultTraceInfo ctor should be enough. It would make the DefaultTraceInfo class substantially larger, but = that's a worthwhile tradeoff if performance is sufficiently improved. = I'll try it out.On the other had, DefaultTraceInfo could be made a struct and be moved =to the Exception class to minimize allocations if you feel that = exceptions should allocate as little as possible (if any at all, because = they may end up throwing OutOfMemory exception).=20 Side question: we don't create OutOfMemory instances, throwing =predefined ones instead, right? I don't think that work with exception = chaining, does it? I mean, what if we e.g. catch an OOM, try to add a = log entry, try to allocate and end up with OOM again. An exception = chaining implemented with an intrusive linked list (i.e. implemented = with an embedded Throwable next;) will not work for this case. Does = druntime handle this case correctly? The general rule for this is that if a static instance is deemed to be = collateral then it's discarded, and if it should replace the current = in-flight exception then the chain is lost. This case used to be = handled explicitly. It isn't any more, but the code still appears to = work (I tested to make sure a few weeks ago). I want to look at this = more carefully though, because I expected to see a segfault and was = surprised when it didn't happen. I don't like unexplained things.=
Apr 26 2011
I'll have to agree that exceptions are quite slow. I was just testing out UTF's decode() function on this UTF-8 test file: http://www.cl.cam.ac.uk/~mgk25/ucs/examples/UTF-8-test.txt . It has a few dozen invalid UTF sequences and is a good stress-test for a decoder. The std.utf.decode function throws an exception on invalid UTF sequences. On that test file (only 20 kilobytes large), loading of a file and skipping invalid sequences takes 210msecs. If I change decode() to use a bool to flag invalid sequences instead of using exceptions, the UTF test file is parsed in 1.4msecs. Now that's quite a difference. Also, loading and validating a 400 kilobyte text file is done in 35msecs, which I'm quite happy about. I'm not sure if the Scintilla editing component applies syntax coloring on the entire file before displaying any lines, but if not then I could easily beat its performance. It takes about a second for it to load and display a 400kbyte source file (with syntax highlighting of course).
May 09 2011
On 5/9/2011 8:09 PM, Andrej Mitrovic wrote:I'll have to agree that exceptions are quite slow. I was just testing out UTF's decode() function on this UTF-8 test file: http://www.cl.cam.ac.uk/~mgk25/ucs/examples/UTF-8-test.txt . It has a few dozen invalid UTF sequences and is a good stress-test for a decoder. The std.utf.decode function throws an exception on invalid UTF sequences. On that test file (only 20 kilobytes large), loading of a file and skipping invalid sequences takes 210msecs. If I change decode() to use a bool to flag invalid sequences instead of using exceptions, the UTF test file is parsed in 1.4msecs. Now that's quite a difference.Validating input data should not be done with exceptions. The exceptions should be thrown only for data that is *expected* to be correct.
May 09 2011
Well I can understand throwing exceptions when using readln() or validate(), but decode() is used for one code point at a time. Throwing is overkill imo.
May 09 2011
On 5/9/2011 9:48 PM, Andrej Mitrovic wrote:Well I can understand throwing exceptions when using readln() or validate(), but decode() is used for one code point at a time. Throwing is overkill imo.Perhaps decode() is badly designed.
May 09 2011
On 5/10/11, Walter Bright <newshound2 digitalmars.com> wrote:On 5/9/2011 9:48 PM, Andrej Mitrovic wrote:It has goto's and exceptions. What's not to like? :pWell I can understand throwing exceptions when using readln() or validate(), but decode() is used for one code point at a time. Throwing is overkill imo.Perhaps decode() is badly designed.
May 09 2011
On 5/10/11, Andrej Mitrovic <andrej.mitrovich gmail.com> wrote:On 5/10/11, Walter Bright <newshound2 digitalmars.com> wrote:Oh and an enforce at the top to kill any potential inlining. Hehehe.On 5/9/2011 9:48 PM, Andrej Mitrovic wrote:It has goto's and exceptions. What's not to like? :pWell I can understand throwing exceptions when using readln() or validate(), but decode() is used for one code point at a time. Throwing is overkill imo.Perhaps decode() is badly designed.
May 09 2011
On Tue, 10 May 2011 01:15:04 -0400, Walter Bright <newshound2 digitalmars.com> wrote:On 5/9/2011 9:48 PM, Andrej Mitrovic wrote:Well, you are supposed to use validate first on any untrusted input. Now the fact that validate returns void and throws an exception and there is no corresponding 'isValid' routine probably is bad design.Well I can understand throwing exceptions when using readln() or validate(), but decode() is used for one code point at a time. Throwing is overkill imo.Perhaps decode() is badly designed.
May 10 2011
On 10.05.2011 16:53, Robert Jacques wrote:Well, you are supposed to use validate first on any untrusted input. Now the fact that validate returns void and throws an exception and there is no corresponding 'isValid' routine probably is bad design.Not to mention that validating first and then running again through the same procedure, thus double-checking input, is a waste of time... ;) /Alexander
May 10 2011
Hi, I'm doing some benchmarks (DMD 2.052 on Linux), and noticed that exception handling is terrible slow even on quite fast hardware. There is a note on DM's site, saying: "Because errors are unusual, execution of error handling code is not performance critical." Well, sure this is (somehow) true, but *so* slow? In my tests, Xeon 3.4 GHz is able to handle only ca. 1000 exceptions/s (!). 1ms for single exception is a little bit too much, especially when application needs to recover fast, not to mention that on slower hardware (like Atom) it will be really slow. And, since some exceptions are not so unusual in normal flow (file not found, connection reset etc), "not performance critical" is not really applicable. Simple example - in case of web (or any other) server with high load, socket exceptions of any kind are quite common. It is still natural to use try/catch to handle them, but it will slow down everything else (imagine > 1K connections/s and 10% cut rate). So, my question - is there something, that can improve performance? Any clues where to dig for this?Yeah. Exceptions are _slow_. I ran into some problems with that when reworking the unit tests in std.datetime. I can barely use assertThrown or it takes the performance of the unit tests. It takes something like 450 times longer to run a line of code that ends up throwing an exception to be caught by assertThrown than it does if it does if a similar call which doesn't thrown that is wrapped by assertNotThrown. So, as far as unit tests are concerned, improving exception performance would be a big boost. Exceptions are bound to be slower, but 450 times slower is an enormous slowdown. If that could be sped up, that would be great. - Jonathan M Davis
Apr 26 2011