type.c
上传用户:blenddy
上传日期:2007-01-07
资源大小:6495k
文件大小:10k
源码类别:

数据库系统

开发平台:

Unix_Linux

  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <stdlib.h>
  4. #include "extern.h"
  5. /* malloc + error check */
  6. void *
  7. mm_alloc(size_t size)
  8. {
  9. void    *ptr = malloc(size);
  10. if (ptr == NULL)
  11. {
  12. fprintf(stderr, "Out of memoryn");
  13. exit(OUT_OF_MEMORY);
  14. }
  15. return ptr;
  16. }
  17. /* strdup + error check */
  18. char *
  19. mm_strdup(const char *string)
  20. {
  21. char    *new = strdup(string);
  22. if (new == NULL)
  23. {
  24. fprintf(stderr, "Out of memoryn");
  25. exit(OUT_OF_MEMORY);
  26. }
  27. return new;
  28. }
  29. /* duplicate memberlist */
  30. struct ECPGstruct_member *
  31. ECPGstruct_member_dup(struct ECPGstruct_member * rm)
  32. {
  33. struct ECPGstruct_member *new = NULL;
  34. while (rm)
  35. {
  36. struct ECPGtype *type;
  37. switch (rm->typ->typ)
  38. {
  39. case ECPGt_struct:
  40. case ECPGt_union:
  41. type = ECPGmake_struct_type(rm->typ->u.members, rm->typ->typ);
  42. break;
  43. case ECPGt_array:
  44. type = ECPGmake_array_type(ECPGmake_simple_type(rm->typ->u.element->typ, rm->typ->u.element->size), rm->typ->size);
  45. break;
  46. default:
  47. type = ECPGmake_simple_type(rm->typ->typ, rm->typ->size);
  48. break;
  49. }
  50. ECPGmake_struct_member(rm->name, type, &new);
  51. rm = rm->next;
  52. }
  53. return (new);
  54. }
  55. /* The NAME argument is copied. The type argument is preserved as a pointer. */
  56. void
  57. ECPGmake_struct_member(char *name, struct ECPGtype * type, struct ECPGstruct_member ** start)
  58. {
  59. struct ECPGstruct_member *ptr,
  60.    *ne =
  61. (struct ECPGstruct_member *) mm_alloc(sizeof(struct ECPGstruct_member));
  62. ne->name = strdup(name);
  63. ne->typ = type;
  64. ne->next = NULL;
  65. for (ptr = *start; ptr && ptr->next; ptr = ptr->next);
  66. if (ptr)
  67. ptr->next = ne;
  68. else
  69. *start = ne;
  70. }
  71. struct ECPGtype *
  72. ECPGmake_simple_type(enum ECPGttype typ, long siz)
  73. {
  74. struct ECPGtype *ne = (struct ECPGtype *) mm_alloc(sizeof(struct ECPGtype));
  75. ne->typ = typ;
  76. ne->size = siz;
  77. ne->u.element = 0;
  78. return ne;
  79. }
  80. struct ECPGtype *
  81. ECPGmake_array_type(struct ECPGtype * typ, long siz)
  82. {
  83. struct ECPGtype *ne = ECPGmake_simple_type(ECPGt_array, siz);
  84. ne->u.element = typ;
  85. return ne;
  86. }
  87. struct ECPGtype *
  88. ECPGmake_struct_type(struct ECPGstruct_member * rm, enum ECPGttype type)
  89. {
  90. struct ECPGtype *ne = ECPGmake_simple_type(type, 1);
  91. ne->u.members = ECPGstruct_member_dup(rm);
  92. return ne;
  93. }
  94. static const char *
  95. get_type(enum ECPGttype typ)
  96. {
  97. switch (typ)
  98. {
  99. case ECPGt_char:
  100. return ("ECPGt_char");
  101. break;
  102. case ECPGt_unsigned_char:
  103. return ("ECPGt_unsigned_char");
  104. break;
  105. case ECPGt_short:
  106. return ("ECPGt_short");
  107. break;
  108. case ECPGt_unsigned_short:
  109. return ("ECPGt_unsigned_short");
  110. break;
  111. case ECPGt_int:
  112. return ("ECPGt_int");
  113. break;
  114. case ECPGt_unsigned_int:
  115. return ("ECPGt_unsigned_int");
  116. break;
  117. case ECPGt_long:
  118. return ("ECPGt_long");
  119. break;
  120. case ECPGt_unsigned_long:
  121. return ("ECPGt_unsigned_int");
  122. break;
  123. case ECPGt_float:
  124. return ("ECPGt_float");
  125. break;
  126. case ECPGt_double:
  127. return ("ECPGt_double");
  128. break;
  129. case ECPGt_bool:
  130. return ("ECPGt_bool");
  131. break;
  132. case ECPGt_varchar:
  133. return ("ECPGt_varchar");
  134. case ECPGt_NO_INDICATOR: /* no indicator */
  135. return ("ECPGt_NO_INDICATOR");
  136. break;
  137. case ECPGt_char_variable: /* string that should not be
  138.  * quoted */
  139. return ("ECPGt_char_variable");
  140. break;
  141. default:
  142. sprintf(errortext, "illegal variable type %dn", typ);
  143. yyerror(errortext);
  144. }
  145. return NULL;
  146. }
  147. /* Dump a type.
  148.    The type is dumped as:
  149.    type-tag <comma>    - enum ECPGttype
  150.    reference-to-variable <comma>   - void *
  151.    size <comma>    - long size of this field (if varchar)
  152.    arrsize <comma>    - long number of elements in the arr
  153.    offset <comma>    - offset to the next element
  154.    Where:
  155.    type-tag is one of the simple types or varchar.
  156.    reference-to-variable can be a reference to a struct element.
  157.    arrsize is the size of the array in case of array fetches. Otherwise 0.
  158.    size is the maxsize in case it is a varchar. Otherwise it is the size of
  159.    the variable (required to do array fetches of structs).
  160.  */
  161. static void ECPGdump_a_simple(FILE *o, const char *name, enum ECPGttype typ,
  162.   long varcharsize,
  163.   long arrsiz, const char *siz, const char *prefix);
  164. static void ECPGdump_a_struct(FILE *o, const char *name, const char *ind_name, long arrsiz,
  165.   struct ECPGtype * typ, struct ECPGtype * ind_typ, const char *offset, const char *prefix, const char *ind_prefix);
  166. void
  167. ECPGdump_a_type(FILE *o, const char *name, struct ECPGtype * typ, const char *ind_name, struct ECPGtype * ind_typ, const char *prefix, const char *ind_prefix)
  168. {
  169. if (ind_typ == NULL)
  170. {
  171. ind_typ = &ecpg_no_indicator;
  172. ind_name = "no_indicator";
  173. }
  174. switch (typ->typ)
  175. {
  176. case ECPGt_array:
  177. switch (typ->u.element->typ)
  178. {
  179. case ECPGt_array:
  180. yyerror("No nested arrays allowed (except strings)"); /* array of array */
  181. break;
  182. case ECPGt_struct:
  183. case ECPGt_union:
  184. ECPGdump_a_struct(o, name, ind_name, typ->size, typ->u.element, ind_typ->u.element, NULL, prefix, ind_prefix);
  185. break;
  186. default:
  187. if (!IS_SIMPLE_TYPE(typ->u.element->typ))
  188. yyerror("Internal error: unknown datatype, please inform pgsql-bugs@postgresql.org");
  189. ECPGdump_a_simple(o, name, typ->u.element->typ,
  190.   typ->u.element->size, typ->size, NULL, prefix);
  191. if (ind_typ->typ == ECPGt_NO_INDICATOR)
  192. ECPGdump_a_simple(o, ind_name, ind_typ->typ, ind_typ->size, -1, NULL, ind_prefix);
  193. else
  194. {
  195. if (ind_typ->typ != ECPGt_array)
  196. {
  197. fprintf(stderr, "Indicator for an array has to be array too.n");
  198. exit(INDICATOR_NOT_ARRAY);
  199. }
  200. ECPGdump_a_simple(o, ind_name, ind_typ->u.element->typ,
  201.   ind_typ->u.element->size, ind_typ->size, NULL, prefix);
  202. }
  203. }
  204. break;
  205. case ECPGt_struct:
  206. ECPGdump_a_struct(o, name, ind_name, 1, typ, ind_typ, NULL, prefix, ind_prefix);
  207. break;
  208. case ECPGt_union: /* cannot dump a complete union */
  209. yyerror("Type of union has to be specified");
  210. break;
  211. case ECPGt_char_variable:
  212. ECPGdump_a_simple(o, name, typ->typ, 1, 1, NULL, prefix);
  213. ECPGdump_a_simple(o, ind_name, ind_typ->typ, ind_typ->size, -1, NULL, ind_prefix);
  214. break;
  215. default:
  216. ECPGdump_a_simple(o, name, typ->typ, typ->size, -1, NULL, prefix);
  217. ECPGdump_a_simple(o, ind_name, ind_typ->typ, ind_typ->size, -1, NULL, ind_prefix);
  218. break;
  219. }
  220. }
  221. /* If siz is NULL, then the offset is 0, if not use siz as a
  222.    string, it represents the offset needed if we are in an array of structs. */
  223. static void
  224. ECPGdump_a_simple(FILE *o, const char *name, enum ECPGttype typ,
  225.   long varcharsize,
  226.   long arrsize,
  227.   const char *siz,
  228.   const char *prefix
  229. )
  230. {
  231. if (typ == ECPGt_NO_INDICATOR)
  232. fprintf(o, "ntECPGt_NO_INDICATOR, NULL , 0L, 0L, 0L, ");
  233. else
  234. {
  235. char    *variable = (char *) mm_alloc(strlen(name) + ((prefix == NULL) ? 0 : strlen(prefix)) + 4);
  236. char    *offset = (char *) mm_alloc(strlen(name) + strlen("sizeof(struct varchar_)") + 1);
  237. /* we have to use the pointer except for arrays with given bounds */
  238. if (arrsize > 0)
  239. sprintf(variable, "(%s%s)", prefix ? prefix : "", name);
  240. else
  241. sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
  242. switch (typ)
  243. {
  244. case ECPGt_varchar:
  245. sprintf(offset, "sizeof(struct varchar_%s)", name);
  246. break;
  247. case ECPGt_char:
  248. case ECPGt_unsigned_char:
  249. case ECPGt_char_variable:
  250. sprintf(offset, "%ld*sizeof(char)", varcharsize == 0 ? 1 : varcharsize);
  251. break;
  252. default:
  253. sprintf(offset, "sizeof(%s)", ECPGtype_name(typ));
  254. break;
  255. }
  256. if (arrsize < 0)
  257. arrsize = 1;
  258. if (siz == NULL)
  259. fprintf(o, "nt%s,%s,%ldL,%ldL,%s, ", get_type(typ), variable, varcharsize, arrsize, offset);
  260. else
  261. fprintf(o, "nt%s,%s,%ldL,%ldL,%s, ", get_type(typ), variable, varcharsize, arrsize, siz);
  262. free(variable);
  263. free(offset);
  264. }
  265. }
  266. /* Penetrate a struct and dump the contents. */
  267. static void
  268. ECPGdump_a_struct(FILE *o, const char *name, const char *ind_name, long arrsiz, struct ECPGtype * typ, struct ECPGtype * ind_typ, const char *offsetarg, const char *prefix, const char *ind_prefix)
  269. {
  270. /*
  271.  * If offset is NULL, then this is the first recursive level. If not
  272.  * then we are in a struct in a struct and the offset is used as
  273.  * offset.
  274.  */
  275. struct ECPGstruct_member *p,
  276.    *ind_p = NULL;
  277. char obuf[BUFSIZ];
  278. char pbuf[BUFSIZ],
  279. ind_pbuf[BUFSIZ];
  280. const char *offset;
  281. if (offsetarg == NULL)
  282. {
  283. sprintf(obuf, "sizeof(%s)", name);
  284. offset = obuf;
  285. }
  286. else
  287. offset = offsetarg;
  288. sprintf(pbuf, "%s%s.", prefix ? prefix : "", name);
  289. prefix = pbuf;
  290. sprintf(ind_pbuf, "%s%s.", ind_prefix ? ind_prefix : "", ind_name);
  291. ind_prefix = ind_pbuf;
  292. if (ind_typ != NULL)
  293. ind_p = ind_typ->u.members;
  294. for (p = typ->u.members; p; p = p->next)
  295. {
  296. ECPGdump_a_type(o, p->name, p->typ, (ind_p != NULL) ? ind_p->name : NULL, (ind_p != NULL) ? ind_p->typ : NULL, prefix, ind_prefix);
  297. if (ind_p != NULL)
  298. ind_p = ind_p->next;
  299. }
  300. }
  301. void
  302. ECPGfree_struct_member(struct ECPGstruct_member * rm)
  303. {
  304. while (rm)
  305. {
  306. struct ECPGstruct_member *p = rm;
  307. rm = rm->next;
  308. free(p->name);
  309. free(p->typ);
  310. free(p);
  311. }
  312. }
  313. void
  314. ECPGfree_type(struct ECPGtype * typ)
  315. {
  316. if (!IS_SIMPLE_TYPE(typ->typ))
  317. {
  318. switch (typ->typ)
  319. {
  320. case ECPGt_array:
  321. switch (typ->u.element->typ)
  322. {
  323. case ECPGt_array:
  324. yyerror("internal error, found multi-dimensional arrayn");
  325. break;
  326. case ECPGt_struct:
  327. case ECPGt_union:
  328. /* Array of structs. */
  329. ECPGfree_struct_member(typ->u.element->u.members);
  330. free(typ->u.members);
  331. break;
  332. default:
  333. if (!IS_SIMPLE_TYPE(typ->u.element->typ))
  334. yyerror("Internal error: unknown datatype, please inform pgsql-bugs@postgresql.org");
  335. free(typ->u.element);
  336. }
  337. break;
  338. case ECPGt_struct:
  339. case ECPGt_union:
  340. ECPGfree_struct_member(typ->u.members);
  341. free(typ->u.members);
  342. break;
  343. default:
  344. sprintf(errortext, "illegal variable type %dn", typ->typ);
  345. yyerror(errortext);
  346. break;
  347. }
  348. }
  349. free(typ);
  350. }