DataType.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include "DataType.h"
  5. #define ARRAY_NAME 256
  6. static int typeNameIndex = 0;
  7. static int typeNameSwap = 0;
  8. static char typeName[2][ARRAY_NAME];
  9. static bool useGlobals = false;
  10. static Structs globalStructs;
  11. const char* vtGetName(ValueType vt) {
  12. switch(vt) {
  13. case VT_INVALID: return "invalid";
  14. case VT_NOT_SET: return "?";
  15. case VT_INT: return "int";
  16. case VT_FLOAT: return "float";
  17. case VT_POINTER: return "pointer";
  18. case VT_ARRAY: return "array";
  19. case VT_LAST: return "invalid";
  20. }
  21. return "invalid";
  22. }
  23. #define VALUE_TYPE_MASK ((1u << VALUE_TYPE_BIT_SIZE) - 1u)
  24. #define VALUE_MAX_OFFSET ((1 << (32 - VALUE_TYPE_BIT_SIZE)) - 1)
  25. ValueType vGetType(Value v) {
  26. uint32 type = v.typeAndOffset & VALUE_TYPE_MASK;
  27. if(type >= VT_LAST) {
  28. return VT_INVALID;
  29. }
  30. return (ValueType)type;
  31. }
  32. bool vSetType(Value* v, ValueType vt) {
  33. if(v == NULL || vt < 0 || vt >= VT_LAST) {
  34. return true;
  35. }
  36. v->typeAndOffset = (v->typeAndOffset & ~VALUE_TYPE_MASK) | vt;
  37. return false;
  38. }
  39. int32 vGetOffset(Value v) {
  40. return (int32)(v.typeAndOffset >> VALUE_TYPE_BIT_SIZE);
  41. }
  42. bool vSetOffset(Value* v, int32 offset) {
  43. if(v == NULL || offset < 0 || offset > VALUE_MAX_OFFSET) {
  44. return true;
  45. }
  46. v->typeAndOffset = (v->typeAndOffset & VALUE_TYPE_MASK) |
  47. ((uint32)offset << VALUE_TYPE_BIT_SIZE);
  48. return false;
  49. }
  50. static void dtAppend(const char* s) {
  51. int index = 0;
  52. while(typeNameIndex < (ARRAY_NAME - 1) && s[index] != '\0') {
  53. typeName[typeNameSwap][typeNameIndex] = s[index];
  54. index++;
  55. typeNameIndex++;
  56. }
  57. typeName[typeNameSwap][typeNameIndex] = '\0';
  58. }
  59. const char* dtGetName(const Structs* sts, DataType dt) {
  60. typeNameSwap = !typeNameSwap;
  61. typeNameIndex = 0;
  62. switch(dtGetType(dt)) {
  63. case RDT_INT: dtAppend("int"); break;
  64. case RDT_FLOAT: dtAppend("float"); break;
  65. case RDT_STRUCT: {
  66. Struct* s = dtGetStruct(sts, dt);
  67. if(s != NULL) {
  68. dtAppend(s->name);
  69. } else {
  70. dtAppend("?-struct");
  71. }
  72. break;
  73. }
  74. case RDT_VOID: dtAppend("void"); break;
  75. case RDT_INVALID: dtAppend("invalid"); break;
  76. default: dtAppend("?");
  77. }
  78. if(dtIsArray(dt)) {
  79. dtAppend("[]");
  80. }
  81. if(dtIsPointer(dt)) {
  82. dtAppend("*");
  83. }
  84. return typeName[typeNameSwap];
  85. }
  86. bool dtGetSize(DataType dt, const Structs* sts, int* size) {
  87. if(dtIsPointer(dt)) {
  88. *size = 1;
  89. return false;
  90. }
  91. switch(dtGetType(dt)) {
  92. case RDT_INT: *size = 1; return false;
  93. case RDT_FLOAT: *size = 1; return false;
  94. case RDT_STRUCT: {
  95. *size = 0;
  96. Struct* st = dtGetStruct(sts, dt);
  97. if(st == NULL) {
  98. return true;
  99. }
  100. for(int i = 0; i < st->amount; i++) {
  101. int add = 0;
  102. if(dtGetSize(st->vars[i].type, sts, &add)) {
  103. return true;
  104. }
  105. *size += add;
  106. }
  107. return false;
  108. }
  109. case RDT_LAST:
  110. case RDT_VOID:
  111. case RDT_INVALID: return true;
  112. }
  113. return true;
  114. }
  115. static DataType dtBuild(RawDataType rdt, bool isPointer, bool isArray,
  116. int32 structId) {
  117. if(rdt < 0 || rdt >= RDT_LAST || structId < 0 ||
  118. structId > DT_MAX_STRUCT_ID) {
  119. return (DataType){RDT_INVALID};
  120. }
  121. return (DataType){((uint32)rdt << DT_TYPE_OFFSET) |
  122. ((uint32)isPointer << DT_IS_POINTER_OFFSET) |
  123. ((uint32)isArray << DT_IS_ARRAY_OFFSET) |
  124. ((uint32)structId << DT_STRUCT_ID_OFFSET)};
  125. }
  126. DataType dtInt(void) {
  127. return dtBuild(RDT_INT, false, false, 0);
  128. }
  129. DataType dtFloat(void) {
  130. return dtBuild(RDT_FLOAT, false, false, 0);
  131. }
  132. DataType dtVoid(void) {
  133. return dtBuild(RDT_VOID, false, false, 0);
  134. }
  135. DataType dtStruct(const Struct* st) {
  136. return dtBuild(RDT_STRUCT, false, false, st->id);
  137. }
  138. DataType dtText(void) {
  139. return dtBuild(RDT_INT, false, true, 0);
  140. }
  141. DataType dtToArray(DataType dt) {
  142. dt.data |= 1 << DT_IS_ARRAY_OFFSET;
  143. return dt;
  144. }
  145. DataType dtRemoveArray(DataType dt) {
  146. dt.data &= ~(((1u << DT_IS_ARRAY_BIT_SIZE) - 1u) << DT_IS_ARRAY_OFFSET);
  147. return dt;
  148. }
  149. DataType dtToPointer(DataType dt) {
  150. dt.data |= 1 << DT_IS_POINTER_OFFSET;
  151. return dt;
  152. }
  153. bool dtRemovePointer(DataType* dt) {
  154. if(dtIsPointer(*dt)) {
  155. dt->data &=
  156. ~(((1u << DT_IS_POINTER_BIT_SIZE) - 1u) << DT_IS_POINTER_OFFSET);
  157. return true;
  158. }
  159. return false;
  160. }
  161. RawDataType dtGetType(DataType dt) {
  162. uint32 type = dt.data & (((1u << DT_TYPE_BIT_SIZE) - 1u) << DT_TYPE_OFFSET);
  163. if(type >= RDT_LAST) {
  164. return RDT_INVALID;
  165. }
  166. return (RawDataType)type;
  167. }
  168. bool dtCompare(DataType a, DataType b) {
  169. return a.data == b.data;
  170. }
  171. bool dtIsInt(DataType dt) {
  172. return dtCompare(dt, dtInt());
  173. }
  174. bool dtIsFloat(DataType dt) {
  175. return dtCompare(dt, dtFloat());
  176. }
  177. bool dtIsVoid(DataType dt) {
  178. return dtCompare(dt, dtVoid());
  179. }
  180. bool dtIsArray(DataType dt) {
  181. return dt.data &
  182. (((1u << DT_IS_ARRAY_BIT_SIZE) - 1u) << DT_IS_ARRAY_OFFSET);
  183. }
  184. bool dtIsStruct(DataType dt) {
  185. return dtGetType(dt) == RDT_STRUCT;
  186. }
  187. bool dtIsPointer(DataType dt) {
  188. return dt.data &
  189. (((1u << DT_IS_POINTER_BIT_SIZE) - 1u) << DT_IS_POINTER_OFFSET);
  190. }
  191. Struct* dtGetStruct(const Structs* sts, DataType dt) {
  192. if(!dtIsStruct(dt)) {
  193. return NULL;
  194. }
  195. int32 structId = (int32)((dt.data >> DT_STRUCT_ID_OFFSET) &
  196. ((1u << DT_STRUCT_ID_BIT_SIZE) - 1u));
  197. if(structId & 1) {
  198. if(!useGlobals) {
  199. return NULL;
  200. }
  201. structId >>= 1;
  202. if(structId < 0 || structId >= globalStructs.entries) {
  203. return NULL;
  204. }
  205. return globalStructs.data + structId;
  206. }
  207. structId >>= 1;
  208. if(structId < 0 || structId >= sts->entries) {
  209. return NULL;
  210. }
  211. return sts->data + structId;
  212. }
  213. void stAddVariable(Struct* st, const char* name, DataType type) {
  214. int index = st->amount;
  215. st->amount++;
  216. st->vars = (StructVariable*)realloc(st->vars, sizeof(StructVariable) *
  217. (size_t)st->amount);
  218. st->vars[index].name = name;
  219. st->vars[index].type = type;
  220. }
  221. void stsInit(Structs* sts) {
  222. sts->capacity = 4;
  223. sts->entries = 0;
  224. sts->data = (Struct*)malloc(sizeof(Struct) * (size_t)sts->capacity);
  225. }
  226. void stsDelete(Structs* sts) {
  227. for(int i = 0; i < sts->entries; i++) {
  228. free(sts->data[i].vars);
  229. }
  230. free(sts->data);
  231. }
  232. static Struct* stsInternSearch(Structs* sts, const char* name) {
  233. for(int i = 0; i < sts->entries; i++) {
  234. if(strcmp(sts->data[i].name, name) == 0) {
  235. return sts->data + i;
  236. }
  237. }
  238. return NULL;
  239. }
  240. Struct* stsSearch(Structs* sts, const char* name) {
  241. if(useGlobals) {
  242. Struct* st = stsInternSearch(&globalStructs, name);
  243. if(st != NULL) {
  244. return st;
  245. }
  246. }
  247. return stsInternSearch(sts, name);
  248. }
  249. Struct* stsAdd(Structs* sts, const char* name) {
  250. if(sts->entries >= sts->capacity) {
  251. sts->capacity *= 2;
  252. sts->data =
  253. (Struct*)realloc(sts->data, sizeof(Struct) * (size_t)sts->capacity);
  254. }
  255. uint16 index = (uint16)sts->entries++;
  256. sts->data[index].id = (uint16)(index << 1);
  257. sts->data[index].amount = 0;
  258. sts->data[index].name = name;
  259. sts->data[index].vars = NULL;
  260. return sts->data + index;
  261. }
  262. void gstsInit(void) {
  263. stsInit(&globalStructs);
  264. useGlobals = true;
  265. }
  266. void gstsDelete(void) {
  267. stsDelete(&globalStructs);
  268. useGlobals = false;
  269. }
  270. Structs* gstsGet(void) {
  271. return &globalStructs;
  272. }
  273. Struct* gstsAdd(const char* name) {
  274. Struct* st = stsAdd(&globalStructs, name);
  275. st->id |= 1;
  276. return st;
  277. }