| 1 | // Macros to try and make declaring and using exceptions easier | 
|---|
| 2 | // No, these are not part of the language, they replace the virtual system. | 
|---|
| 3 |  | 
|---|
| 4 | // Internal use: | 
|---|
| 5 | #define GLUE2(left, right) left##right | 
|---|
| 6 | #define GLUE3(left, middle, right) left##middle##right | 
|---|
| 7 |  | 
|---|
| 8 | // The fully (perhaps overly) qualified name of the base exception type: | 
|---|
| 9 | #define BASE_EXCEPT __cfaabi_ehm__base_exception_t | 
|---|
| 10 |  | 
|---|
| 11 | // Get the name of the vtable type and vtable instance for an exception type: | 
|---|
| 12 | #define TABLE(name) GLUE2(name,_vtable) | 
|---|
| 13 | #define INSTANCE(name) GLUE3(_,name,_vtable_instance) | 
|---|
| 14 |  | 
|---|
| 15 | // Throws and the bit of overhead: | 
|---|
| 16 | #define THROW(expr) throw ((BASE_EXCEPT *)(expr)) | 
|---|
| 17 | #define THROW_RESUME(expr) throwResume ((BASE_EXCEPT *)(expr)) | 
|---|
| 18 |  | 
|---|
| 19 |  | 
|---|
| 20 |  | 
|---|
| 21 | // The following macros are for defining your own new exception types. | 
|---|
| 22 |  | 
|---|
| 23 | // Declare vtable and forward declare the exception type and vtable instance. | 
|---|
| 24 | // This should start a new exception declaration. | 
|---|
| 25 | // ... argument is the additional vtable fields. | 
|---|
| 26 | #define DECLARE_EXCEPT(except_name,parent_name,...) \ | 
|---|
| 27 | struct except_name; \ | 
|---|
| 28 | struct TABLE(except_name) { \ | 
|---|
| 29 | struct TABLE(parent_name) const * parent; \ | 
|---|
| 30 | size_t size; \ | 
|---|
| 31 | void (*copy)(except_name *this, except_name * other); \ | 
|---|
| 32 | void (*free)(except_name &this); \ | 
|---|
| 33 | const char * (*msg)(except_name *this); \ | 
|---|
| 34 | __VA_ARGS__ \ | 
|---|
| 35 | }; \ | 
|---|
| 36 | extern TABLE(except_name) INSTANCE(except_name); | 
|---|
| 37 |  | 
|---|
| 38 | // The first field of the exception structure should be created with this. | 
|---|
| 39 | #define VTABLE_FIELD(except_name) \ | 
|---|
| 40 | struct TABLE(except_name) const * virtual_table | 
|---|
| 41 |  | 
|---|
| 42 | // In each constructor the vtable must be initialized. | 
|---|
| 43 | #define VTABLE_INIT(this_name,except_name) \ | 
|---|
| 44 | this_name.virtual_table = &INSTANCE(except_name) | 
|---|
| 45 |  | 
|---|
| 46 | // Declare the vtable instance. This should end an exception declaration. | 
|---|
| 47 | // ... argument is the remaining vtable field values. | 
|---|
| 48 | #define VTABLE_INSTANCE(except_name,parent_name,copy,free,msg,...) \ | 
|---|
| 49 | TABLE(except_name) INSTANCE(except_name) @= { \ | 
|---|
| 50 | &INSTANCE(parent_name), sizeof(except_name), \ | 
|---|
| 51 | copy, free, msg, ## __VA_ARGS__ \ | 
|---|
| 52 | }; | 
|---|
| 53 |  | 
|---|
| 54 | // Same, but used declarators for arguments. | 
|---|
| 55 | #define VTABLE_INSTANCE_KEY(except_name,parent_name,copy,free,msg,...) \ | 
|---|
| 56 | TABLE(except_name) INSTANCE(except_name) @= { \ | 
|---|
| 57 | .parent : &INSTANCE(parent_name), .size : sizeof(except_name), \ | 
|---|
| 58 | .copy : copy, .free : free, .msg : msg, ## __VA_ARGS__ \ | 
|---|
| 59 | }; | 
|---|
| 60 |  | 
|---|
| 61 |  | 
|---|
| 62 |  | 
|---|
| 63 | // Declare a trivial exception, one that adds no features: | 
|---|
| 64 | #define TRIVIAL_EXCEPTION(name) \ | 
|---|
| 65 | DECLARE_EXCEPT(name,BASE_EXCEPT,) \ | 
|---|
| 66 | struct name { \ | 
|---|
| 67 | VTABLE_FIELD(name); \ | 
|---|
| 68 | }; \ | 
|---|
| 69 | const char * GLUE2(name,_msg)(name * this) { \ | 
|---|
| 70 | return #name; \ | 
|---|
| 71 | } \ | 
|---|
| 72 | void GLUE2(name,_copy)(name * this, name * other) { \ | 
|---|
| 73 | this->virtual_table = other->virtual_table; \ | 
|---|
| 74 | } \ | 
|---|
| 75 | void ?{}(name & this) { \ | 
|---|
| 76 | VTABLE_INIT(this,name); \ | 
|---|
| 77 | } \ | 
|---|
| 78 | VTABLE_INSTANCE(name,BASE_EXCEPT,GLUE2(name,_copy),^?{},GLUE2(name,_msg),) | 
|---|