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),)
|
---|