2023-10-11 19:01:06 +00:00
|
|
|
// C reflection: enums, functions, structs, members and anotations.
|
|
|
|
// - rlyeh, public domain
|
|
|
|
//
|
|
|
|
// @todo: nested structs? pointers in members?
|
|
|
|
// @todo: declare TYPEDEF(vec3, float[3]), TYPEDEF(mat4, vec4[4]/*float[16]*/)
|
|
|
|
|
2023-10-16 15:15:59 +00:00
|
|
|
#ifndef OBJTYPE
|
|
|
|
#define OBJTYPE(T) 0
|
2023-10-13 10:59:44 +00:00
|
|
|
#endif
|
2023-10-11 19:01:06 +00:00
|
|
|
|
2023-10-15 11:16:35 +00:00
|
|
|
typedef struct reflect_t {
|
2023-10-11 19:01:06 +00:00
|
|
|
unsigned id, objtype;
|
2023-10-16 20:07:29 +00:00
|
|
|
union {
|
2023-10-11 19:01:06 +00:00
|
|
|
unsigned sz;
|
2023-10-16 20:07:29 +00:00
|
|
|
unsigned member_offset;
|
|
|
|
unsigned enum_value;
|
|
|
|
};
|
2023-10-11 19:01:06 +00:00
|
|
|
const char *name;
|
|
|
|
const char *info;
|
|
|
|
void *addr;
|
|
|
|
unsigned parent;
|
|
|
|
const char *type;
|
2023-10-16 20:07:29 +00:00
|
|
|
unsigned bytes;
|
2023-10-15 11:16:35 +00:00
|
|
|
} reflect_t;
|
2023-10-11 19:01:06 +00:00
|
|
|
|
|
|
|
// inscribe api
|
|
|
|
|
2023-10-16 20:07:29 +00:00
|
|
|
#define ENUM(V, .../*value_annotations*/) \
|
|
|
|
enum_inscribe(#V,intern(#V),V, "" __VA_ARGS__/*value_annotations*/)
|
2023-10-11 19:01:06 +00:00
|
|
|
|
2023-10-16 20:07:29 +00:00
|
|
|
#define FUNCTION(F, .../*function_annotations*/) \
|
|
|
|
function_inscribe(#F,intern(#F),(void*)F, "" __VA_ARGS__/*function_annotations*/)
|
2023-10-11 19:01:06 +00:00
|
|
|
|
2023-10-16 20:07:29 +00:00
|
|
|
#define STRUCT(T, type, member, .../*member_annotations*/) \
|
2023-10-16 15:15:59 +00:00
|
|
|
struct_inscribe(#T,intern(#T),sizeof(T),OBJTYPE(T),NULL), \
|
2023-10-16 20:07:29 +00:00
|
|
|
type_inscribe(#type,intern(#type),sizeof(((T){0}).member),"" __VA_ARGS__/*member_annotations*/), \
|
|
|
|
member_inscribe(intern(#T), #member,intern(#member),(uintptr_t)&((T*)0)->member, "" __VA_ARGS__/*member_annotations*/, #type, sizeof(((T){0}).member) )
|
2023-10-11 19:01:06 +00:00
|
|
|
|
|
|
|
// find api
|
|
|
|
|
|
|
|
API unsigned enum_find(const char *E);
|
|
|
|
API void * function_find(const char *F);
|
|
|
|
|
2023-10-15 11:16:35 +00:00
|
|
|
API reflect_t member_find(const char *T, const char *M); /// find specific member
|
2023-10-16 20:07:29 +00:00
|
|
|
API void * member_findptr(void *obj, const char *T, const char *M); // @deprecate
|
2023-10-15 11:16:35 +00:00
|
|
|
API array(reflect_t) members_find(const char *T);
|
2023-10-11 19:01:06 +00:00
|
|
|
|
|
|
|
// iterate members in a struct
|
|
|
|
|
|
|
|
#define each_member(T,R) \
|
2023-10-15 11:16:35 +00:00
|
|
|
(array(reflect_t)*found_ = map_find(members, intern(T)); found_; found_ = 0) \
|
2023-10-11 19:01:06 +00:00
|
|
|
for(int it_ = 0, end_ = array_count(*found_); it_ != end_; ++it_ ) \
|
2023-10-15 11:16:35 +00:00
|
|
|
for(reflect_t *R = (*found_)+it_; R; R = 0 )
|
2023-10-11 19:01:06 +00:00
|
|
|
|
|
|
|
// private api, still exposed
|
|
|
|
|
|
|
|
API void type_inscribe(const char *TY,unsigned TYid,unsigned TYsz,const char *infos);
|
|
|
|
API void enum_inscribe(const char *E,unsigned Eid,unsigned Eval,const char *infos);
|
|
|
|
API void struct_inscribe(const char *T,unsigned Tid,unsigned Tsz,unsigned OBJTYPEid, const char *infos);
|
2023-10-16 20:07:29 +00:00
|
|
|
API void member_inscribe(unsigned Tid, const char *M,unsigned Mid,unsigned Msz, const char *infos, const char *type, unsigned bytes);
|
2023-10-11 19:01:06 +00:00
|
|
|
API void function_inscribe(const char *F,unsigned Fid,void *func,const char *infos);
|
|
|
|
|
2023-10-16 15:15:59 +00:00
|
|
|
API void reflect_print(const char *symbol);
|
|
|
|
API void reflect_dump(const char *mask);
|
|
|
|
API void reflect_init();
|