gs_meta.h 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. /*==============================================================================================================
  2. * Copyright (c) 2020 John Jackson
  3. * GSMeta: Meta Data Util for Gunslinger
  4. * File: gs_meta.h
  5. * Github: https://github.com/MrFrenik/gunslinger
  6. * All Rights Reserved
  7. * MIT License
  8. * May all those that this source may reach be blessed by the LORD and find peace and joy in life.
  9. * Everyone who drinks of this water will be thirsty again; but whoever drinks of the water
  10. * that I will give him shall never thirst; John 4:13-14
  11. * Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
  12. * documentation files (the "Software"), to deal in the Software without restriction, including without limitation
  13. * the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software,
  14. * and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
  15. * The above copyright, blessing, biblical verse, notice and this permission notice shall be included in all
  16. * copies or substantial portions of the Software.
  17. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
  18. * TO THE WARRANTIES OF MECHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
  19. * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
  20. * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  21. * IN THE SOFTWARE.
  22. =================================================================================================================*/
  23. #ifndef GS_META_H
  24. #define GS_META_H
  25. /*
  26. USAGE: (IMPORTANT)
  27. =================================================================================================================
  28. Before including, define the gunslinger asset manager implementation like this:
  29. #define GS_META_IMPL
  30. in EXACTLY ONE C or C++ file that includes this header, BEFORE the
  31. include, like this:
  32. #define GS_META_IMPL
  33. #include "gs_meta.h"
  34. All other files should just #include "gs_meta.h" without the #define.
  35. MUST include "gs.h" and declare GS_IMPL BEFORE this file, since this file relies on that:
  36. #define GS_IMPL
  37. #include "gs.h"
  38. #define GS_META_IMPL
  39. #include "gs_meta.h"
  40. ================================================================================================================
  41. */
  42. /*==== Interface ====*/
  43. /** @defgroup gs_meta_util Meta Data Util
  44. * Gunslinger Meta Data Util
  45. * @{
  46. */
  47. #define GS_META_PROPERTY_FLAG_POINTER 0x01
  48. #define GS_META_PROPERTY_FLAG_DOUBLE_POINTER 0x02
  49. typedef struct gs_meta_property_type_info_t
  50. {
  51. const char* name; // Used for display name
  52. uint32_t id; // Id used for lookups and associations (most likely by enum)
  53. uint32_t flags; // Number of times this field needs to be dereferenced
  54. union
  55. {
  56. struct
  57. {
  58. uint64_t enum_id;
  59. } enum_info;
  60. struct
  61. {
  62. uint32_t key_id;
  63. uint32_t val_id;
  64. } container_info;
  65. } info;
  66. } gs_meta_property_type_info_t;
  67. // Default meta property type ids
  68. typedef enum gs_meta_property_type
  69. {
  70. GS_META_PROPERTY_TYPE_U8 = 0x00,
  71. GS_META_PROPERTY_TYPE_U16,
  72. GS_META_PROPERTY_TYPE_U32,
  73. GS_META_PROPERTY_TYPE_U64,
  74. GS_META_PROPERTY_TYPE_S8,
  75. GS_META_PROPERTY_TYPE_S16,
  76. GS_META_PROPERTY_TYPE_S32,
  77. GS_META_PROPERTY_TYPE_S64,
  78. GS_META_PROPERTY_TYPE_F32,
  79. GS_META_PROPERTY_TYPE_F64,
  80. GS_META_PROPERTY_TYPE_ENUM,
  81. GS_META_PROPERTY_TYPE_VEC2,
  82. GS_META_PROPERTY_TYPE_VEC3,
  83. GS_META_PROPERTY_TYPE_VEC4,
  84. GS_META_PROPERTY_TYPE_QUAT,
  85. GS_META_PROPERTY_TYPE_MAT3,
  86. GS_META_PROPERTY_TYPE_MAT4,
  87. GS_META_PROPERTY_TYPE_VQS,
  88. GS_META_PROPERTY_TYPE_UUID,
  89. GS_META_PROPERTY_TYPE_SIZE_T, // Used for pointers or size_t variables
  90. GS_META_PROPERTY_TYPE_STR, // Used for const char*, char*
  91. GS_META_PROPERTY_TYPE_COLOR,
  92. GS_META_PROPERTY_TYPE_OBJ,
  93. GS_META_PROPERTY_TYPE_COUNT
  94. } gs_meta_property_type;
  95. GS_API_PRIVATE gs_meta_property_type_info_t _gs_meta_property_type_decl_impl(const char* name, uint32_t id);
  96. #define _gs_meta_property_type_decl(T, PROP_TYPE)\
  97. _gs_meta_property_type_decl_impl(gs_to_str(T), PROP_TYPE)
  98. // Default meta property type info defines
  99. #define GS_META_PROPERTY_TYPE_INFO_U8 _gs_meta_property_type_decl(uint8_t, GS_META_PROPERTY_TYPE_U8)
  100. #define GS_META_PROPERTY_TYPE_INFO_S8 _gs_meta_property_type_decl(int8_t, GS_META_PROPERTY_TYPE_S8)
  101. #define GS_META_PROPERTY_TYPE_INFO_U16 _gs_meta_property_type_decl(uint16_t, GS_META_PROPERTY_TYPE_U16)
  102. #define GS_META_PROPERTY_TYPE_INFO_S16 _gs_meta_property_type_decl(int16_t, GS_META_PROPERTY_TYPE_S16)
  103. #define GS_META_PROPERTY_TYPE_INFO_U32 _gs_meta_property_type_decl(uint32_t, GS_META_PROPERTY_TYPE_U32)
  104. #define GS_META_PROPERTY_TYPE_INFO_S32 _gs_meta_property_type_decl(int32_t, GS_META_PROPERTY_TYPE_S32)
  105. #define GS_META_PROPERTY_TYPE_INFO_U64 _gs_meta_property_type_decl(uint64_t, GS_META_PROPERTY_TYPE_U64)
  106. #define GS_META_PROPERTY_TYPE_INFO_S64 _gs_meta_property_type_decl(int64_t, GS_META_PROPERTY_TYPE_S64)
  107. #define GS_META_PROPERTY_TYPE_INFO_F32 _gs_meta_property_type_decl(float, GS_META_PROPERTY_TYPE_F32)
  108. #define GS_META_PROPERTY_TYPE_INFO_F64 _gs_meta_property_type_decl(double, GS_META_PROPERTY_TYPE_F64)
  109. #define GS_META_PROPERTY_TYPE_INFO_ENUM _gs_meta_property_type_decl(enum, GS_META_PROPERTY_TYPE_ENUM)
  110. #define GS_META_PROPERTY_TYPE_INFO_VEC2 _gs_meta_property_type_decl(gs_vec2, GS_META_PROPERTY_TYPE_VEC2)
  111. #define GS_META_PROPERTY_TYPE_INFO_VEC3 _gs_meta_property_type_decl(gs_vec3, GS_META_PROPERTY_TYPE_VEC3)
  112. #define GS_META_PROPERTY_TYPE_INFO_VEC4 _gs_meta_property_type_decl(gs_vec4, GS_META_PROPERTY_TYPE_VEC4)
  113. #define GS_META_PROPERTY_TYPE_INFO_QUAT _gs_meta_property_type_decl(gs_quat, GS_META_PROPERTY_TYPE_QUAT)
  114. #define GS_META_PROPERTY_TYPE_INFO_MAT3 _gs_meta_property_type_decl(gs_mat3, GS_META_PROPERTY_TYPE_MAT3)
  115. #define GS_META_PROPERTY_TYPE_INFO_MAT4 _gs_meta_property_type_decl(gs_mat4, GS_META_PROPERTY_TYPE_MAT4)
  116. #define GS_META_PROPERTY_TYPE_INFO_VQS _gs_meta_property_type_decl(gs_vqs, GS_META_PROPERTY_TYPE_VQS)
  117. #define GS_META_PROPERTY_TYPE_INFO_UUID _gs_meta_property_type_decl(gs_uuid_t, GS_META_PROPERTY_TYPE_UUID)
  118. #define GS_META_PROPERTY_TYPE_INFO_SIZE_T _gs_meta_property_type_decl(size_t, GS_META_PROPERTY_TYPE_SIZE_T)
  119. #define GS_META_PROPERTY_TYPE_INFO_STR _gs_meta_property_type_decl(char*, GS_META_PROPERTY_TYPE_STR)
  120. #define GS_META_PROPERTY_TYPE_INFO_COLOR _gs_meta_property_type_decl(gs_color_t, GS_META_PROPERTY_TYPE_COLOR)
  121. #define GS_META_PROPERTY_TYPE_INFO_OBJ _gs_meta_property_type_decl(object, GS_META_PROPERTY_TYPE_OBJ)
  122. typedef struct gs_meta_property_t
  123. {
  124. const char* name;
  125. uint32_t offset;
  126. const char* type_name;
  127. size_t size;
  128. gs_meta_property_type_info_t type;
  129. } gs_meta_property_t;
  130. typedef struct gs_meta_enum_value_t
  131. {
  132. const char* name;
  133. } gs_meta_enum_value_t;
  134. GS_API_PRIVATE gs_meta_property_t _gs_meta_property_impl(const char* field_type_name, const char* field, size_t sz, uint32_t offset, gs_meta_property_type_info_t type);
  135. #define gs_meta_property(CLS, FIELD_TYPE, FIELD, TYPE)\
  136. _gs_meta_property_impl(gs_to_str(FIELD_TYPE), gs_to_str(FIELD),\
  137. sizeof(FIELD_TYPE), gs_offset(CLS, FIELD), TYPE)
  138. typedef struct gs_meta_vtable_t
  139. {
  140. gs_hash_table(uint64_t, void*) funcs; // Hash function name to function pointer
  141. } gs_meta_vtable_t;
  142. typedef struct gs_meta_class_t
  143. {
  144. gs_meta_property_t* properties; // Property list
  145. uint32_t property_count; // Number of properties in list
  146. gs_hash_table(uint64_t, gs_meta_property_t*) property_map; // Mapping of hash property name to pointer
  147. const char* name; // Display name of class
  148. uint64_t id; // Class ID
  149. uint64_t base; // Parent class ID
  150. gs_meta_vtable_t vtable; // VTable for class
  151. size_t size; // Size of class in bytes (for heap allocations)
  152. } gs_meta_class_t;
  153. typedef struct gs_meta_enum_t
  154. {
  155. gs_meta_enum_value_t* values; // Value list
  156. uint32_t value_count; // Count of enum values
  157. const char* name; // Name of enum
  158. uint64_t id; // Enum id
  159. } gs_meta_enum_t;
  160. typedef struct gs_meta_registry_t
  161. {
  162. gs_hash_table(uint64_t, gs_meta_class_t) classes;
  163. gs_hash_table(uint64_t, gs_meta_enum_t) enums;
  164. void* user_data;
  165. } gs_meta_registry_t;
  166. typedef struct gs_meta_class_decl_t
  167. {
  168. gs_meta_property_t* properties;
  169. size_t size;
  170. const char* name; // Display name of class
  171. const char* base; // Base parent class name (will be used for hash id, NULL for invalid id)
  172. gs_meta_vtable_t* vtable; // Vtable
  173. size_t cls_size; // Size of class in bytes
  174. } gs_meta_class_decl_t;
  175. typedef struct gs_meta_enum_decl_t
  176. {
  177. gs_meta_enum_value_t* values;
  178. size_t size;
  179. const char* name; // Display name of class
  180. } gs_meta_enum_decl_t;
  181. GS_API_DECL gs_meta_registry_t gs_meta_registry_new();
  182. GS_API_DECL void gs_meta_registry_free(gs_meta_registry_t* meta);
  183. GS_API_DECL const char* gs_meta_typestr(gs_meta_property_type type);
  184. GS_API_DECL bool32 gs_meta_has_base_class(const gs_meta_registry_t* meta, const gs_meta_class_t* cls);
  185. GS_API_DECL uint64_t gs_meta_class_register(gs_meta_registry_t* meta, const gs_meta_class_decl_t* decl);
  186. GS_API_DECL void gs_meta_class_unregister(gs_meta_registry_t* meta, uint64_t id);
  187. GS_API_DECL uint64_t gs_meta_enum_register(gs_meta_registry_t* meta, const gs_meta_enum_decl_t* decl);
  188. #define gs_meta_class_get(META, T)\
  189. (gs_hash_table_getp((META)->classes, gs_hash_str64(gs_to_str(T))))
  190. #define gs_meta_class_get_w_name(META, NAME)\
  191. (gs_hash_table_getp((META)->classes, gs_hash_str64(NAME)))
  192. #define gs_meta_class_get_w_id(META, ID)\
  193. (gs_hash_table_getp((META)->classes, (ID)))
  194. #define gs_meta_class_exists(META, ID)\
  195. (gs_hash_table_exists((META)->classes, ID))
  196. #define gs_meta_getv(OBJ, T, PROP)\
  197. (*((T*)((uint8_t*)(OBJ) + (PROP)->offset)))
  198. #define gs_meta_getvp(OBJ, T, PROP)\
  199. (((T*)((uint8_t*)(OBJ) + (PROP)->offset)))
  200. #define gs_meta_setv(OBJ, T, PROP, VAL)\
  201. (*((T*)((uint8_t*)(OBJ) + (PROP)->offset)) = VAL)
  202. #define gs_meta_func_get(CLS, NAME)\
  203. (_gs_meta_func_get_internal(CLS, gs_to_str(NAME)));
  204. #define gs_meta_func_get_w_id(META, ID, NAME)\
  205. (_gs_meta_func_get_internal_w_id(META, ID, gs_to_str(NAME)));
  206. GS_API_DECL void* _gs_meta_func_get_internal(const gs_meta_class_t* cls, const char* func_name);
  207. GS_API_DECL void* _gs_meta_func_get_internal_w_id(const gs_meta_registry_t* meta, uint64_t id, const char* func_name);
  208. // Reflection Utils
  209. /** @} */ // end of gs_meta_data_util
  210. /*==== Implementation ====*/
  211. #ifdef GS_META_IMPL
  212. GS_API_DECL gs_meta_registry_t gs_meta_registry_new()
  213. {
  214. gs_meta_registry_t meta = gs_default_val();
  215. return meta;
  216. }
  217. GS_API_DECL void gs_meta_registry_free(gs_meta_registry_t* meta)
  218. {
  219. // Free all entries in classes
  220. for (
  221. gs_hash_table_iter it = gs_hash_table_iter_new(meta->classes);
  222. gs_hash_table_iter_valid(meta->classes, it);
  223. gs_hash_table_iter_advance(meta->classes, it)
  224. )
  225. {
  226. gs_meta_class_t* cls = gs_hash_table_iter_getp(meta->classes, it);
  227. gs_free(cls->properties);
  228. }
  229. gs_hash_table_free(meta->classes);
  230. }
  231. GS_API_PRIVATE gs_meta_property_t
  232. _gs_meta_property_impl(const char* field_type_name, const char* field, size_t size,
  233. uint32_t offset, gs_meta_property_type_info_t type)
  234. {
  235. gs_meta_property_t mp = gs_default_val();
  236. mp.name = field;
  237. mp.type_name = field_type_name;
  238. mp.offset = offset;
  239. mp.size = size;
  240. mp.type = type;
  241. return mp;
  242. }
  243. GS_API_PRIVATE uint64_t
  244. gs_meta_class_register(gs_meta_registry_t* meta, const gs_meta_class_decl_t* decl)
  245. {
  246. uint32_t ct = decl->size / sizeof(gs_meta_property_t);
  247. gs_meta_class_t cls = gs_default_val();
  248. cls.properties = (gs_meta_property_t*)gs_malloc(decl->size);
  249. cls.property_count = ct;
  250. memcpy(cls.properties, decl->properties, decl->size);
  251. for (uint32_t i = 0; i < cls.property_count; ++i)
  252. {
  253. gs_meta_property_t* prop = &cls.properties[i];
  254. gs_hash_table_insert(cls.property_map, gs_hash_str64(prop->name), prop);
  255. }
  256. cls.name = decl->name;
  257. cls.base = decl->base ? gs_hash_str64(decl->base) : gs_hash_str64("NULL");
  258. uint64_t id = gs_hash_str64(decl->name);
  259. cls.id = id;
  260. cls.vtable = decl->vtable ? *decl->vtable : cls.vtable;
  261. cls.size = decl->cls_size;
  262. gs_hash_table_insert(meta->classes, id, cls);
  263. return id;
  264. }
  265. GS_API_DECL void
  266. gs_meta_class_unregister(gs_meta_registry_t* meta, uint64_t id)
  267. {
  268. gs_meta_class_t* cls = gs_hash_table_getp(meta->classes, id);
  269. if (cls->properties) gs_free(cls->properties);
  270. if (cls->property_map) gs_hash_table_free(cls->property_map);
  271. gs_hash_table_erase(meta->classes, id);
  272. }
  273. GS_API_DECL uint64_t
  274. gs_meta_enum_register(gs_meta_registry_t* meta, const gs_meta_enum_decl_t* decl)
  275. {
  276. uint32_t ct = decl->size / sizeof(gs_meta_enum_value_t);
  277. gs_meta_enum_t enm = gs_default_val();
  278. enm.values = (gs_meta_enum_value_t*)gs_malloc(decl->size);
  279. enm.value_count = ct;
  280. enm.name = decl->name;
  281. memcpy(enm.values, decl->values, decl->size);
  282. uint64_t id = gs_hash_str64(decl->name);
  283. enm.id = id;
  284. gs_hash_table_insert(meta->enums, id, enm);
  285. return id;
  286. }
  287. GS_API_PRIVATE gs_meta_property_type_info_t _gs_meta_property_type_decl_impl(const char* name, uint32_t id)
  288. {
  289. gs_meta_property_type_info_t info = gs_default_val();
  290. info.name = name;
  291. info.id = id;
  292. return info;
  293. }
  294. GS_API_DECL bool32 gs_meta_has_base_class(const gs_meta_registry_t* meta, const gs_meta_class_t* cls)
  295. {
  296. return (gs_hash_table_key_exists(meta->classes, cls->base));
  297. }
  298. GS_API_DECL void* _gs_meta_func_get_internal(const gs_meta_class_t* cls, const char* func_name)
  299. {
  300. uint64_t hash = gs_hash_str64(func_name);
  301. if (gs_hash_table_exists(cls->vtable.funcs, hash))
  302. {
  303. return gs_hash_table_get(cls->vtable.funcs, hash);
  304. }
  305. return NULL;
  306. }
  307. GS_API_DECL void* _gs_meta_func_get_internal_w_id(const gs_meta_registry_t* meta, uint64_t id, const char* func_name)
  308. {
  309. const gs_meta_class_t* cls = gs_hash_table_getp(meta->classes, id);
  310. uint64_t hash = gs_hash_str64(func_name);
  311. if (gs_hash_table_exists(cls->vtable.funcs, hash))
  312. {
  313. return gs_hash_table_get(cls->vtable.funcs, hash);
  314. }
  315. return NULL;
  316. }
  317. #undef GS_META_IMP
  318. #endif // GS_META_IMPL
  319. #endif // GS_META_H