![]() |
|
4.3 error handling |
Kitty Cat
Member #2,815
October 2002
![]() |
Because SF is giving problems with the mailing list, and I could always use more feedback, I thought I'd toss up my idea about error handling in 4.3 here. My initial email follows (with [code] blocks added for your sanity My original message said: While playing around with the sound code again, I thought it'd be nice if it had some error handling abilities. I came up with this in about 15 to 30 minutes. I was just wondering if this is a good direction to put it in, or if you wanted something else. Here's the main code:
C99 defines __thread, and Windows has a similar specifier. So as you can see, it uses TLS, meaning seperate threads erroring won't conflict with each other. To use it, you have these methods: ** Method #1 (the lazy/newbie approach): sound_driver = al_audio_init_driver(NULL); /* Allegro will have SIGABRT'd by now if there was a problem, we can continue assuming everything's okay */ ..more game code.. ** Method #2 (the C++ approach): al_error_try ( sound_driver = al_audio_init_driver(NULL); ..more related code.. } al_error_catch { AL_ERROR_ENUM code = al_get_last_error(); /* We only handle invalid parameter errors */ if(code != AL_INVALID_PARAM) al_error_throw(code); show_my_error("Could not init sound!"); } ..more game code.. ** Method #3 (the C approach): al_error_disable_throws(); sound_driver = al_audio_init_driver(NULL); if(!sound_driver) { AL_ERROR_ENUM code = al_get_last_error(); if(code != AL_INVALID_PARAM) { al_error_enable_throws(); al_error_throw(code); } show_my_error("Could not init sound!"); } al_error_enable_throws(); ..more game code.. So, what do you think? I kinda like it, but it may be too hacky for some people's tastes. I believe it's logically sound though, so it shouldn't cause problems.. After this, Elias responded: His response said: I think, for C code, those #defines are really ugly.. at least to me they do look ugly. If I wanted to use #2, then I would use C++, not C. And actual C++ users wouldn't benefit from setjmp/longjmp and those #defines, since real exceptions should be raised for C++. I'd say, the best idea for C++ is a wrapper, where al_sound::init would raise an exception. The TLS error code and NULL return is all that seems needed to me for C. For the #1 approach, we could have a flag to al_init, like al_init(ABORT_ON_ERROR). And then Allegro would automatically abort on error - just like in your code. The #3 approach looks much simpler than the #2 approach to me anyway, so I don't think anything is wrong with it One idea similar to setjmp/longjmp might be an error callback, something like: To which I responded: My response said: [quote He]I think, for C code, those #defines are really ugly.. at least to me they do look ugly. If I wanted to use #2, then I would use C++, not C. But Allegro doesn't use C++. Since Allegro is going to use C (AFAIK), or at least expose a C-only API, it wouldn't be able to properly throw real exceptions. Besides, with the way that code is, you can use any of the three at any time. Don't want to use the try/catch defines? Then just call al_error_disable_throws() once at the beginning of your program and don't think anything of them (just be sure to check for failure return codes). Too lazy to do your own error checking? Leave throws enabled and don't worry about the try and catch defines. There's many times I wish I could just execute a string of commands and not have to check the return code of each one individually. But I'd still like to be able to handle the errors manually myself, without having to use a seperate callback function. Something like:
Having to check each function in that try block manually and basically call the same error-handling code would be very wasteful. And putting those few lines into a seperate function to install as a callback temporarilly would not be very clean (plus the trouble of being able to save and restore a previous callback, if one was installed, and optionally continue from outside of the "error zone"). NOTE: The al_get_error_string function I threw in up there isn't in the code I posted, but it would be useful to have a human-readable string of the problem along with a simple error code. He said: And actual C++ users wouldn't benefit from setjmp/longjmp and those #defines, since real exceptions should be raised for C++. I'd say, the best idea for C++ is a wrapper, where al_sound::init would raise an exception. A C++ wrapper could still raise real exceptions: al_audio::init() { al_error_try { driver = al_audio_init_driver(NULL); } al_error_catch { throw al_get_last_error(); } } That also has the benefit of leaving the real throwing in C++, so you needn't have to worry about C++ exceptions through C. He said: The TLS error code and NULL return is all that seems needed to me for C. Then that's all you have to use. Just call al_error_disable_throws() and do it the C way. He said: For the #1 approach, we could have a flag to al_init, like al_init(ABORT_ON_ERROR). And then Allegro would automatically abort on error - just like in your code. My code has the benefit of being able to turn such aborts on and off at will. Like, say, if you want to init video without doing specific error checking, but then check the errors while initializing sound. It's also transparent, so you could put in a single try/catch around the whole initialization section without having to change the initialization code itself. He said:
The #3 approach looks much simpler than the #2 approach to me anyway, so I don't think anything is wrong with it
Then, again, you're free to use it. He said: One idea similar to setjmp/longjmp might be an error callback, something like: void al_set_error_callback(void *(user_error_handler)(AL_ERROR error)); It might be a good idea to use that anyway, for uncaught errors (eg. if an error is generated and either throws are disabled, or not within a try block). The error handler can then return non-0 to tell Allegro to clean up and SIGABRT, or 0 to tell it to keep going. Though it probably shouldn't be allowed to keep going if throws are enabled. He said: Or they could simply have abort in there, then we wouldn't even need an extra flag for it in al_init.
Allegro knows best how to clean itself up (as long as it can keep track of its That said, however, I do see a small problem with my code. You wouldn't be -- |
Ron Novy
Member #6,982
March 2006
![]() |
Here is a simple, thread safe and easy to use C way that I believe might be a better compromise.
Any way check it out, compile it then let me know what you think. [EDIT] I attached a new version that does a lot of ascii->current text format conversions. I'm not sure if they would be necessary but it works just as well. ---- |
kazzmir
Member #1,786
December 2001
![]() |
I dont really see how that helps. What if you want to perform some cleanup at the end of a block of code? How can you emulate that by putting a TRY{} thing around each statement? The point is to put all of the error code in one place. |
Kitty Cat
Member #2,815
October 2002
![]() |
There seems to be a question over whether setjmp/longjmp is actually safe to use with local variables lying around. As I understand it, local variables are supposed to be on the stack, and can be put into registers for speed optimizations. Since they're supposed to be on the stack, I'd think the compiler would be smart enough to not do anything dangerous around setjmp/longjmp that would compromise that (eg. copying vars from registers into the proper stack location before longjmp completes, and having the vars reloaded from the stack into the proper registers before setjmp returns non-0). Does anyone have any info that might clear this up? Also another idea might be to include a C++ exception throwing callback for errors (eg. an error occurs, Allegro calls a callback, etc), and have an al_init option to use it by default. However, the issue with that seems to be thus:
</li> -- |
Matthew Leverton
Supreme Loser
January 1999
![]() |
I've always been suspicious of libraries that try to over extend the language into something it's not. I think if someone really wants exception handling, they should use C++ and a wrapper over Allegro that provides that. And yes, I realize that doesn't help out people writing core Allegro routines. But if your proposed system did work on every compiler under all situations, I wouldn't see any harm in it. I'd still think it didn't belong, but I'm not the one writing code... |
Peter Wang
Member #23
April 2000
|
Quote: There seems to be a question over whether setjmp/longjmp is actually safe to use with local variables lying around. It's not a question. Try this program: On my system, compiling with `gcc -O0' prints "0". Compiling with `gcc -O' or above prints "1". You really do need to declare `c' as `volatile'.
|
Carrus85
Member #2,633
August 2002
![]() |
Of course, if allegro were to use c++, this whole chain of arguments would be a nonissue. (Or better yet, the core allegro code is in C, then a C++ wrapper is written to interact with the "core" (Depending on design, one could even write a C wrapper as well that abstracts some of the technicalities away from the core)). So, you basically make the core library require as verbose of arguments as possible (don't assume things based on previous settings and such unless the function's nature requires it), and write wrappers to abstract things away from the complicated core library (so, for example, in the core library, you might only have one version of create_bitmap, that takes width, height, and color depth arguments. The wrapper library can handle global colordepths externally to the core library).
|
Ron Novy
Member #6,982
March 2006
![]() |
I'm not sure if setjump is thread-safe though. Imagine that multiple threads tried to use setjump at the same time. The results would then be undefined. After my previous post I searched through allegro.c and discovered that my code was very similar to allegro's ASSERT code. Maybe there could be a solution that would jump using a #define macro instead of calling a function directly.
[edit] ---- |
Thomas Fjellstrom
Member #476
June 2000
![]() |
Quote: I'm not sure if setjump is thread-safe though. Which is why the OP suggested TLS (thread local storage, aka __thread) allong with set/longjmp. -- |
Kitty Cat
Member #2,815
October 2002
![]() |
One of the things I was going for was a clean code execution path. So you could run some code, if it errors, stop and run some cleanup code, then continue running code. eg: al_audio_init(); al_create_voices(); if there was an error { al_audio_close(); warn("no sound!"); } more_code_here(); Of course, if the first function fails, you don't want to run the second. But in either case, you want to run the same error handling code, then continue on whether there was an error or not. As well, for newbies or someone who just wants to quickly get something down, you're not going to want or know to tediously check every function for errors, so having a simple way to automatically catch errors for you, at least in the beginning, would be very helpful. -- |
|