LCOV - code coverage report
Current view: top level - src/interfaces/ecpg/preproc - type.c (source / functions) Coverage Total Hit
Test: Code coverage Lines: 0.0 % 349 0
Test Date: 2026-01-26 10:56:24 Functions: 0.0 % 12 0
Legend: Lines:     hit not hit

            Line data    Source code
       1              : /* src/interfaces/ecpg/preproc/type.c */
       2              : 
       3              : #include "postgres_fe.h"
       4              : 
       5              : #include "preproc_extern.h"
       6              : 
       7              : #define indicator_set ind_type != NULL && ind_type->type != ECPGt_NO_INDICATOR
       8              : 
       9              : static struct ECPGstruct_member struct_no_indicator = {"no_indicator", &ecpg_no_indicator, NULL};
      10              : 
      11              : /* duplicate memberlist */
      12              : struct ECPGstruct_member *
      13            0 : ECPGstruct_member_dup(struct ECPGstruct_member *rm)
      14              : {
      15            0 :         struct ECPGstruct_member *new = NULL;
      16              : 
      17            0 :         while (rm)
      18              :         {
      19            0 :                 struct ECPGtype *type;
      20              : 
      21            0 :                 switch (rm->type->type)
      22              :                 {
      23              :                         case ECPGt_struct:
      24              :                         case ECPGt_union:
      25            0 :                                 type = ECPGmake_struct_type(rm->type->u.members, rm->type->type, rm->type->type_name, rm->type->struct_sizeof);
      26            0 :                                 break;
      27              :                         case ECPGt_array:
      28              : 
      29              :                                 /*
      30              :                                  * if this array does contain a struct again, we have to
      31              :                                  * create the struct too
      32              :                                  */
      33            0 :                                 if (rm->type->u.element->type == ECPGt_struct || rm->type->u.element->type == ECPGt_union)
      34            0 :                                         type = ECPGmake_struct_type(rm->type->u.element->u.members, rm->type->u.element->type, rm->type->u.element->type_name, rm->type->u.element->struct_sizeof);
      35              :                                 else
      36            0 :                                         type = ECPGmake_array_type(ECPGmake_simple_type(rm->type->u.element->type, rm->type->u.element->size, rm->type->u.element->counter), rm->type->size);
      37            0 :                                 break;
      38              :                         default:
      39            0 :                                 type = ECPGmake_simple_type(rm->type->type, rm->type->size, rm->type->counter);
      40            0 :                                 break;
      41              :                 }
      42              : 
      43            0 :                 ECPGmake_struct_member(rm->name, type, &new);
      44              : 
      45            0 :                 rm = rm->next;
      46            0 :         }
      47              : 
      48            0 :         return new;
      49            0 : }
      50              : 
      51              : /* The NAME argument is copied. The type argument is preserved as a pointer. */
      52              : void
      53            0 : ECPGmake_struct_member(const char *name, struct ECPGtype *type, struct ECPGstruct_member **start)
      54              : {
      55            0 :         struct ECPGstruct_member *ptr,
      56            0 :                            *ne =
      57            0 :                 (struct ECPGstruct_member *) mm_alloc(sizeof(struct ECPGstruct_member));
      58              : 
      59            0 :         ne->name = mm_strdup(name);
      60            0 :         ne->type = type;
      61            0 :         ne->next = NULL;
      62              : 
      63            0 :         for (ptr = *start; ptr && ptr->next; ptr = ptr->next);
      64              : 
      65            0 :         if (ptr)
      66            0 :                 ptr->next = ne;
      67              :         else
      68            0 :                 *start = ne;
      69            0 : }
      70              : 
      71              : struct ECPGtype *
      72            0 : ECPGmake_simple_type(enum ECPGttype type, const char *size, int counter)
      73              : {
      74            0 :         struct ECPGtype *ne = (struct ECPGtype *) mm_alloc(sizeof(struct ECPGtype));
      75              : 
      76            0 :         ne->type = type;
      77            0 :         ne->type_name = NULL;
      78            0 :         ne->size = mm_strdup(size);
      79            0 :         ne->u.element = NULL;
      80            0 :         ne->struct_sizeof = NULL;
      81            0 :         ne->counter = counter;               /* only needed for varchar and bytea */
      82              : 
      83            0 :         return ne;
      84            0 : }
      85              : 
      86              : struct ECPGtype *
      87            0 : ECPGmake_array_type(struct ECPGtype *type, const char *size)
      88              : {
      89            0 :         struct ECPGtype *ne = ECPGmake_simple_type(ECPGt_array, size, 0);
      90              : 
      91            0 :         ne->u.element = type;
      92              : 
      93            0 :         return ne;
      94            0 : }
      95              : 
      96              : struct ECPGtype *
      97            0 : ECPGmake_struct_type(struct ECPGstruct_member *rm, enum ECPGttype type,
      98              :                                          const char *type_name, const char *struct_sizeof)
      99              : {
     100            0 :         struct ECPGtype *ne = ECPGmake_simple_type(type, "1", 0);
     101              : 
     102            0 :         ne->type_name = mm_strdup(type_name);
     103            0 :         ne->u.members = ECPGstruct_member_dup(rm);
     104            0 :         ne->struct_sizeof = mm_strdup(struct_sizeof);
     105              : 
     106            0 :         return ne;
     107            0 : }
     108              : 
     109              : static const char *
     110            0 : get_type(enum ECPGttype type)
     111              : {
     112            0 :         switch (type)
     113              :         {
     114              :                 case ECPGt_char:
     115            0 :                         return "ECPGt_char";
     116              :                         break;
     117              :                 case ECPGt_unsigned_char:
     118            0 :                         return "ECPGt_unsigned_char";
     119              :                         break;
     120              :                 case ECPGt_short:
     121            0 :                         return "ECPGt_short";
     122              :                         break;
     123              :                 case ECPGt_unsigned_short:
     124            0 :                         return "ECPGt_unsigned_short";
     125              :                         break;
     126              :                 case ECPGt_int:
     127            0 :                         return "ECPGt_int";
     128              :                         break;
     129              :                 case ECPGt_unsigned_int:
     130            0 :                         return "ECPGt_unsigned_int";
     131              :                         break;
     132              :                 case ECPGt_long:
     133            0 :                         return "ECPGt_long";
     134              :                         break;
     135              :                 case ECPGt_unsigned_long:
     136            0 :                         return "ECPGt_unsigned_long";
     137              :                         break;
     138              :                 case ECPGt_long_long:
     139            0 :                         return "ECPGt_long_long";
     140              :                         break;
     141              :                 case ECPGt_unsigned_long_long:
     142            0 :                         return "ECPGt_unsigned_long_long";
     143              :                         break;
     144              :                 case ECPGt_float:
     145            0 :                         return "ECPGt_float";
     146              :                         break;
     147              :                 case ECPGt_double:
     148            0 :                         return "ECPGt_double";
     149              :                         break;
     150              :                 case ECPGt_bool:
     151            0 :                         return "ECPGt_bool";
     152              :                         break;
     153              :                 case ECPGt_varchar:
     154            0 :                         return "ECPGt_varchar";
     155              :                 case ECPGt_bytea:
     156            0 :                         return "ECPGt_bytea";
     157              :                 case ECPGt_NO_INDICATOR:        /* no indicator */
     158            0 :                         return "ECPGt_NO_INDICATOR";
     159              :                         break;
     160              :                 case ECPGt_char_variable:       /* string that should not be quoted */
     161            0 :                         return "ECPGt_char_variable";
     162              :                         break;
     163              :                 case ECPGt_const:               /* constant string quoted */
     164            0 :                         return "ECPGt_const";
     165              :                         break;
     166              :                 case ECPGt_decimal:
     167            0 :                         return "ECPGt_decimal";
     168              :                         break;
     169              :                 case ECPGt_numeric:
     170            0 :                         return "ECPGt_numeric";
     171              :                         break;
     172              :                 case ECPGt_interval:
     173            0 :                         return "ECPGt_interval";
     174              :                         break;
     175              :                 case ECPGt_descriptor:
     176            0 :                         return "ECPGt_descriptor";
     177              :                         break;
     178              :                 case ECPGt_sqlda:
     179            0 :                         return "ECPGt_sqlda";
     180              :                         break;
     181              :                 case ECPGt_date:
     182            0 :                         return "ECPGt_date";
     183              :                         break;
     184              :                 case ECPGt_timestamp:
     185            0 :                         return "ECPGt_timestamp";
     186              :                         break;
     187              :                 case ECPGt_string:
     188            0 :                         return "ECPGt_string";
     189              :                         break;
     190              :                 default:
     191            0 :                         mmerror(PARSE_ERROR, ET_ERROR, "unrecognized variable type code %d", type);
     192            0 :         }
     193              : 
     194            0 :         return NULL;
     195            0 : }
     196              : 
     197              : /* Dump a type.
     198              :    The type is dumped as:
     199              :    type-tag <comma>                                  - enum ECPGttype
     200              :    reference-to-variable <comma>             - char *
     201              :    size <comma>                                      - long size of this field (if varchar)
     202              :    arrsize <comma>                                   - long number of elements in the arr
     203              :    offset <comma>                            - offset to the next element
     204              :    Where:
     205              :    type-tag is one of the simple types or varchar.
     206              :    reference-to-variable can be a reference to a struct element.
     207              :    arrsize is the size of the array in case of array fetches. Otherwise 0.
     208              :    size is the maxsize in case it is a varchar. Otherwise it is the size of
     209              :    the variable (required to do array fetches of structs).
     210              :  */
     211              : static void ECPGdump_a_simple(FILE *o, const char *name, enum ECPGttype type,
     212              :                                                           char *varcharsize,
     213              :                                                           char *arrsize, const char *size, const char *prefix, int counter);
     214              : static void ECPGdump_a_struct(FILE *o, const char *name, const char *ind_name, char *arrsize,
     215              :                                                           struct ECPGtype *type, struct ECPGtype *ind_type, const char *prefix, const char *ind_prefix);
     216              : 
     217              : void
     218            0 : ECPGdump_a_type(FILE *o, const char *name, struct ECPGtype *type, const int brace_level,
     219              :                                 const char *ind_name, struct ECPGtype *ind_type, const int ind_brace_level,
     220              :                                 const char *prefix, const char *ind_prefix,
     221              :                                 char *arr_str_size, const char *struct_sizeof,
     222              :                                 const char *ind_struct_sizeof)
     223              : {
     224            0 :         struct variable *var;
     225              : 
     226            0 :         if (type->type != ECPGt_descriptor && type->type != ECPGt_sqlda &&
     227            0 :                 type->type != ECPGt_char_variable && type->type != ECPGt_const &&
     228            0 :                 brace_level >= 0)
     229              :         {
     230            0 :                 char       *str;
     231              : 
     232            0 :                 str = mm_strdup(name);
     233            0 :                 var = find_variable(str);
     234            0 :                 free(str);
     235              : 
     236            0 :                 if ((var->type->type != type->type) ||
     237            0 :                         (var->type->type_name && !type->type_name) ||
     238            0 :                         (!var->type->type_name && type->type_name) ||
     239            0 :                         (var->type->type_name && type->type_name && strcmp(var->type->type_name, type->type_name) != 0))
     240            0 :                         mmerror(PARSE_ERROR, ET_ERROR, "variable \"%s\" is hidden by a local variable of a different type", name);
     241            0 :                 else if (var->brace_level != brace_level)
     242            0 :                         mmerror(PARSE_ERROR, ET_WARNING, "variable \"%s\" is hidden by a local variable", name);
     243              : 
     244            0 :                 if (ind_name && ind_type && ind_type->type != ECPGt_NO_INDICATOR && ind_brace_level >= 0)
     245              :                 {
     246            0 :                         str = mm_strdup(ind_name);
     247            0 :                         var = find_variable(str);
     248            0 :                         free(str);
     249              : 
     250            0 :                         if ((var->type->type != ind_type->type) ||
     251            0 :                                 (var->type->type_name && !ind_type->type_name) ||
     252            0 :                                 (!var->type->type_name && ind_type->type_name) ||
     253            0 :                                 (var->type->type_name && ind_type->type_name && strcmp(var->type->type_name, ind_type->type_name) != 0))
     254            0 :                                 mmerror(PARSE_ERROR, ET_ERROR, "indicator variable \"%s\" is hidden by a local variable of a different type", ind_name);
     255            0 :                         else if (var->brace_level != ind_brace_level)
     256            0 :                                 mmerror(PARSE_ERROR, ET_WARNING, "indicator variable \"%s\" is hidden by a local variable", ind_name);
     257            0 :                 }
     258            0 :         }
     259              : 
     260            0 :         switch (type->type)
     261              :         {
     262              :                 case ECPGt_array:
     263            0 :                         if (indicator_set && ind_type->type != ECPGt_array)
     264            0 :                                 mmfatal(INDICATOR_NOT_ARRAY, "indicator for array/pointer has to be array/pointer");
     265            0 :                         switch (type->u.element->type)
     266              :                         {
     267              :                                 case ECPGt_array:
     268            0 :                                         mmerror(PARSE_ERROR, ET_ERROR, "nested arrays are not supported (except strings)"); /* array of array */
     269            0 :                                         break;
     270              :                                 case ECPGt_struct:
     271              :                                 case ECPGt_union:
     272            0 :                                         ECPGdump_a_struct(o, name,
     273            0 :                                                                           ind_name,
     274            0 :                                                                           type->size,
     275            0 :                                                                           type->u.element,
     276            0 :                                                                           (ind_type == NULL) ? NULL : ((ind_type->type == ECPGt_NO_INDICATOR) ? ind_type : ind_type->u.element),
     277            0 :                                                                           prefix, ind_prefix);
     278            0 :                                         break;
     279              :                                 default:
     280            0 :                                         if (!IS_SIMPLE_TYPE(type->u.element->type))
     281            0 :                                                 base_yyerror("internal error: unknown datatype, please report this to <" PACKAGE_BUGREPORT ">");
     282              : 
     283            0 :                                         ECPGdump_a_simple(o, name,
     284            0 :                                                                           type->u.element->type,
     285            0 :                                                                           type->u.element->size, type->size, struct_sizeof ? struct_sizeof : NULL,
     286            0 :                                                                           prefix, type->u.element->counter);
     287              : 
     288            0 :                                         if (ind_type != NULL)
     289              :                                         {
     290            0 :                                                 if (ind_type->type == ECPGt_NO_INDICATOR)
     291              :                                                 {
     292            0 :                                                         char       *str_neg_one = mm_strdup("-1");
     293              : 
     294            0 :                                                         ECPGdump_a_simple(o, ind_name, ind_type->type, ind_type->size, str_neg_one, NULL, ind_prefix, 0);
     295            0 :                                                         free(str_neg_one);
     296            0 :                                                 }
     297              :                                                 else
     298              :                                                 {
     299            0 :                                                         ECPGdump_a_simple(o, ind_name, ind_type->u.element->type,
     300            0 :                                                                                           ind_type->u.element->size, ind_type->size, NULL, ind_prefix, 0);
     301              :                                                 }
     302            0 :                                         }
     303            0 :                         }
     304            0 :                         break;
     305              :                 case ECPGt_struct:
     306              :                         {
     307            0 :                                 char       *str_one = mm_strdup("1");
     308              : 
     309            0 :                                 if (indicator_set && ind_type->type != ECPGt_struct)
     310            0 :                                         mmfatal(INDICATOR_NOT_STRUCT, "indicator for struct has to be a struct");
     311              : 
     312            0 :                                 ECPGdump_a_struct(o, name, ind_name, str_one, type, ind_type, prefix, ind_prefix);
     313            0 :                                 free(str_one);
     314            0 :                         }
     315            0 :                         break;
     316              :                 case ECPGt_union:               /* cannot dump a complete union */
     317            0 :                         base_yyerror("type of union has to be specified");
     318            0 :                         break;
     319              :                 case ECPGt_char_variable:
     320              :                         {
     321              :                                 /*
     322              :                                  * Allocate for each, as there are code-paths where the values
     323              :                                  * get stomped on.
     324              :                                  */
     325            0 :                                 char       *str_varchar_one = mm_strdup("1");
     326            0 :                                 char       *str_arr_one = mm_strdup("1");
     327            0 :                                 char       *str_neg_one = mm_strdup("-1");
     328              : 
     329            0 :                                 if (indicator_set && (ind_type->type == ECPGt_struct || ind_type->type == ECPGt_array))
     330            0 :                                         mmfatal(INDICATOR_NOT_SIMPLE, "indicator for simple data type has to be simple");
     331              : 
     332            0 :                                 ECPGdump_a_simple(o, name, type->type, str_varchar_one, (arr_str_size && strcmp(arr_str_size, "0") != 0) ? arr_str_size : str_arr_one, struct_sizeof, prefix, 0);
     333            0 :                                 if (ind_type != NULL)
     334            0 :                                         ECPGdump_a_simple(o, ind_name, ind_type->type, ind_type->size, (arr_str_size && strcmp(arr_str_size, "0") != 0) ? arr_str_size : str_neg_one, ind_struct_sizeof, ind_prefix, 0);
     335              : 
     336            0 :                                 free(str_varchar_one);
     337            0 :                                 free(str_arr_one);
     338            0 :                                 free(str_neg_one);
     339            0 :                         }
     340            0 :                         break;
     341              :                 case ECPGt_descriptor:
     342              :                         {
     343              :                                 /*
     344              :                                  * Allocate for each, as there are code-paths where the values
     345              :                                  * get stomped on.
     346              :                                  */
     347            0 :                                 char       *str_neg_one = mm_strdup("-1");
     348            0 :                                 char       *ind_type_neg_one = mm_strdup("-1");
     349              : 
     350            0 :                                 if (indicator_set && (ind_type->type == ECPGt_struct || ind_type->type == ECPGt_array))
     351            0 :                                         mmfatal(INDICATOR_NOT_SIMPLE, "indicator for simple data type has to be simple");
     352              : 
     353            0 :                                 ECPGdump_a_simple(o, name, type->type, NULL, str_neg_one, NULL, prefix, 0);
     354            0 :                                 if (ind_type != NULL)
     355            0 :                                         ECPGdump_a_simple(o, ind_name, ind_type->type, ind_type->size, ind_type_neg_one, NULL, ind_prefix, 0);
     356              : 
     357            0 :                                 free(str_neg_one);
     358            0 :                                 free(ind_type_neg_one);
     359            0 :                         }
     360            0 :                         break;
     361              :                 default:
     362              :                         {
     363              :                                 /*
     364              :                                  * Allocate for each, as there are code-paths where the values
     365              :                                  * get stomped on.
     366              :                                  */
     367            0 :                                 char       *str_neg_one = mm_strdup("-1");
     368            0 :                                 char       *ind_type_neg_one = mm_strdup("-1");
     369              : 
     370            0 :                                 if (indicator_set && (ind_type->type == ECPGt_struct || ind_type->type == ECPGt_array))
     371            0 :                                         mmfatal(INDICATOR_NOT_SIMPLE, "indicator for simple data type has to be simple");
     372              : 
     373            0 :                                 ECPGdump_a_simple(o, name, type->type, type->size, (arr_str_size && strcmp(arr_str_size, "0") != 0) ? arr_str_size : str_neg_one, struct_sizeof, prefix, type->counter);
     374            0 :                                 if (ind_type != NULL)
     375            0 :                                         ECPGdump_a_simple(o, ind_name, ind_type->type, ind_type->size, (arr_str_size && strcmp(arr_str_size, "0") != 0) ? arr_str_size : ind_type_neg_one, ind_struct_sizeof, ind_prefix, 0);
     376              : 
     377            0 :                                 free(str_neg_one);
     378            0 :                                 free(ind_type_neg_one);
     379            0 :                         }
     380            0 :                         break;
     381              :         }
     382            0 : }
     383              : 
     384              : 
     385              : /* If size is NULL, then the offset is 0, if not use size as a
     386              :    string, it represents the offset needed if we are in an array of structs. */
     387              : static void
     388            0 : ECPGdump_a_simple(FILE *o, const char *name, enum ECPGttype type,
     389              :                                   char *varcharsize,
     390              :                                   char *arrsize,
     391              :                                   const char *size,
     392              :                                   const char *prefix,
     393              :                                   int counter)
     394              : {
     395            0 :         if (type == ECPGt_NO_INDICATOR)
     396            0 :                 fprintf(o, "\n\tECPGt_NO_INDICATOR, NULL , 0L, 0L, 0L, ");
     397            0 :         else if (type == ECPGt_descriptor)
     398              :                 /* remember that name here already contains quotes (if needed) */
     399            0 :                 fprintf(o, "\n\tECPGt_descriptor, %s, 1L, 1L, 1L, ", name);
     400            0 :         else if (type == ECPGt_sqlda)
     401            0 :                 fprintf(o, "\n\tECPGt_sqlda, &%s, 0L, 0L, 0L, ", name);
     402              :         else
     403              :         {
     404            0 :                 char       *variable = (char *) mm_alloc(strlen(name) + ((prefix == NULL) ? 0 : strlen(prefix)) + 4);
     405            0 :                 char       *offset = (char *) mm_alloc(strlen(name) + strlen("sizeof(struct varchar_)") + 1 + strlen(varcharsize) + sizeof(int) * CHAR_BIT * 10 / 3);
     406            0 :                 char       *struct_name;
     407              : 
     408            0 :                 switch (type)
     409              :                 {
     410              :                                 /*
     411              :                                  * we have to use the & operator except for arrays and
     412              :                                  * pointers
     413              :                                  */
     414              : 
     415              :                         case ECPGt_varchar:
     416              :                         case ECPGt_bytea:
     417              : 
     418              :                                 /*
     419              :                                  * we have to use the pointer except for arrays with given
     420              :                                  * bounds
     421              :                                  */
     422            0 :                                 if (((atoi(arrsize) > 0) ||
     423            0 :                                          (atoi(arrsize) == 0 && strcmp(arrsize, "0") != 0)) &&
     424            0 :                                         size == NULL)
     425            0 :                                         sprintf(variable, "(%s%s)", prefix ? prefix : "", name);
     426              :                                 else
     427            0 :                                         sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
     428              : 
     429              :                                 /*
     430              :                                  * If we created a varchar structure automatically, counter is
     431              :                                  * greater than 0.
     432              :                                  */
     433            0 :                                 if (type == ECPGt_varchar)
     434            0 :                                         struct_name = "struct varchar";
     435              :                                 else
     436            0 :                                         struct_name = "struct bytea";
     437              : 
     438            0 :                                 if (counter)
     439            0 :                                         sprintf(offset, "sizeof(%s_%d)", struct_name, counter);
     440              :                                 else
     441            0 :                                         sprintf(offset, "sizeof(%s)", struct_name);
     442            0 :                                 break;
     443              :                         case ECPGt_char:
     444              :                         case ECPGt_unsigned_char:
     445              :                         case ECPGt_char_variable:
     446              :                         case ECPGt_string:
     447              :                                 {
     448            0 :                                         char       *sizeof_name = "char";
     449              : 
     450              :                                         /*
     451              :                                          * we have to use the pointer except for arrays with given
     452              :                                          * bounds, ecpglib will distinguish between * and []
     453              :                                          */
     454            0 :                                         if ((atoi(varcharsize) > 1 ||
     455            0 :                                                  (atoi(arrsize) > 0) ||
     456            0 :                                                  (atoi(varcharsize) == 0 && strcmp(varcharsize, "0") != 0) ||
     457            0 :                                                  (atoi(arrsize) == 0 && strcmp(arrsize, "0") != 0))
     458            0 :                                                 && size == NULL)
     459              :                                         {
     460            0 :                                                 sprintf(variable, "(%s%s)", prefix ? prefix : "", name);
     461            0 :                                                 if ((type == ECPGt_char || type == ECPGt_unsigned_char) &&
     462            0 :                                                         strcmp(varcharsize, "0") == 0)
     463              :                                                 {
     464              :                                                         /*
     465              :                                                          * If this is an array of char *, the offset would
     466              :                                                          * be sizeof(char *) and not sizeof(char).
     467              :                                                          */
     468            0 :                                                         sizeof_name = "char *";
     469            0 :                                                 }
     470            0 :                                         }
     471              :                                         else
     472            0 :                                                 sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
     473              : 
     474            0 :                                         sprintf(offset, "(%s)*sizeof(%s)", strcmp(varcharsize, "0") == 0 ? "1" : varcharsize, sizeof_name);
     475              :                                         break;
     476            0 :                                 }
     477              :                         case ECPGt_numeric:
     478              : 
     479              :                                 /*
     480              :                                  * we have to use a pointer here
     481              :                                  */
     482            0 :                                 sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
     483            0 :                                 sprintf(offset, "sizeof(numeric)");
     484            0 :                                 break;
     485              :                         case ECPGt_interval:
     486              : 
     487              :                                 /*
     488              :                                  * we have to use a pointer here
     489              :                                  */
     490            0 :                                 sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
     491            0 :                                 sprintf(offset, "sizeof(interval)");
     492            0 :                                 break;
     493              :                         case ECPGt_date:
     494              : 
     495              :                                 /*
     496              :                                  * we have to use a pointer and translate the variable type
     497              :                                  */
     498            0 :                                 sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
     499            0 :                                 sprintf(offset, "sizeof(date)");
     500            0 :                                 break;
     501              :                         case ECPGt_timestamp:
     502              : 
     503              :                                 /*
     504              :                                  * we have to use a pointer and translate the variable type
     505              :                                  */
     506            0 :                                 sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
     507            0 :                                 sprintf(offset, "sizeof(timestamp)");
     508            0 :                                 break;
     509              :                         case ECPGt_const:
     510              : 
     511              :                                 /*
     512              :                                  * just dump the const as string
     513              :                                  */
     514            0 :                                 sprintf(variable, "\"%s\"", name);
     515            0 :                                 sprintf(offset, "strlen(\"%s\")", name);
     516            0 :                                 break;
     517              :                         default:
     518              : 
     519              :                                 /*
     520              :                                  * we have to use the pointer except for arrays with given
     521              :                                  * bounds
     522              :                                  */
     523            0 :                                 if (((atoi(arrsize) > 0) ||
     524            0 :                                          (atoi(arrsize) == 0 && strcmp(arrsize, "0") != 0)) &&
     525            0 :                                         size == NULL)
     526            0 :                                         sprintf(variable, "(%s%s)", prefix ? prefix : "", name);
     527              :                                 else
     528            0 :                                         sprintf(variable, "&(%s%s)", prefix ? prefix : "", name);
     529              : 
     530            0 :                                 sprintf(offset, "sizeof(%s)", ecpg_type_name(type));
     531            0 :                                 break;
     532              :                 }
     533              : 
     534              :                 /*
     535              :                  * Array size would be -1 for addresses of members within structure,
     536              :                  * when pointer to structure is being dumped.
     537              :                  */
     538            0 :                 if (atoi(arrsize) < 0 && !size)
     539            0 :                         strcpy(arrsize, "1");
     540              : 
     541              :                 /*
     542              :                  * If size i.e. the size of structure of which this variable is part
     543              :                  * of, that gives the offset to the next element, if required
     544              :                  */
     545            0 :                 if (size == NULL || strlen(size) == 0)
     546            0 :                         fprintf(o, "\n\t%s,%s,(long)%s,(long)%s,%s, ", get_type(type), variable, varcharsize, arrsize, offset);
     547              :                 else
     548            0 :                         fprintf(o, "\n\t%s,%s,(long)%s,(long)%s,%s, ", get_type(type), variable, varcharsize, arrsize, size);
     549              : 
     550            0 :                 free(variable);
     551            0 :                 free(offset);
     552            0 :         }
     553            0 : }
     554              : 
     555              : 
     556              : /* Penetrate a struct and dump the contents. */
     557              : static void
     558            0 : ECPGdump_a_struct(FILE *o, const char *name, const char *ind_name, char *arrsize, struct ECPGtype *type, struct ECPGtype *ind_type, const char *prefix, const char *ind_prefix)
     559              : {
     560              :         /*
     561              :          * If offset is NULL, then this is the first recursive level. If not then
     562              :          * we are in a struct and the offset is used as offset.
     563              :          */
     564            0 :         struct ECPGstruct_member *p,
     565            0 :                            *ind_p = NULL;
     566            0 :         char       *pbuf = (char *) mm_alloc(strlen(name) + ((prefix == NULL) ? 0 : strlen(prefix)) + 3);
     567            0 :         char       *ind_pbuf = (char *) mm_alloc(strlen(ind_name) + ((ind_prefix == NULL) ? 0 : strlen(ind_prefix)) + 3);
     568              : 
     569            0 :         if (atoi(arrsize) == 1)
     570            0 :                 sprintf(pbuf, "%s%s.", prefix ? prefix : "", name);
     571              :         else
     572            0 :                 sprintf(pbuf, "%s%s->", prefix ? prefix : "", name);
     573              : 
     574            0 :         prefix = pbuf;
     575              : 
     576            0 :         if (ind_type == &ecpg_no_indicator)
     577            0 :                 ind_p = &struct_no_indicator;
     578            0 :         else if (ind_type != NULL)
     579              :         {
     580            0 :                 if (atoi(arrsize) == 1)
     581            0 :                         sprintf(ind_pbuf, "%s%s.", ind_prefix ? ind_prefix : "", ind_name);
     582              :                 else
     583            0 :                         sprintf(ind_pbuf, "%s%s->", ind_prefix ? ind_prefix : "", ind_name);
     584              : 
     585            0 :                 ind_prefix = ind_pbuf;
     586            0 :                 ind_p = ind_type->u.members;
     587            0 :         }
     588              : 
     589            0 :         for (p = type->u.members; p; p = p->next)
     590              :         {
     591            0 :                 ECPGdump_a_type(o, p->name, p->type, -1,
     592            0 :                                                 (ind_p != NULL) ? ind_p->name : NULL,
     593            0 :                                                 (ind_p != NULL) ? ind_p->type : NULL,
     594              :                                                 -1,
     595            0 :                                                 prefix, ind_prefix, arrsize, type->struct_sizeof,
     596            0 :                                                 (ind_p != NULL) ? ind_type->struct_sizeof : NULL);
     597            0 :                 if (ind_p != NULL && ind_p != &struct_no_indicator)
     598              :                 {
     599            0 :                         ind_p = ind_p->next;
     600            0 :                         if (ind_p == NULL && p->next != NULL)
     601              :                         {
     602            0 :                                 mmerror(PARSE_ERROR, ET_WARNING, "indicator struct \"%s\" has too few members", ind_name);
     603            0 :                                 ind_p = &struct_no_indicator;
     604            0 :                         }
     605            0 :                 }
     606            0 :         }
     607              : 
     608            0 :         if (ind_type != NULL && ind_p != NULL && ind_p != &struct_no_indicator)
     609              :         {
     610            0 :                 mmerror(PARSE_ERROR, ET_WARNING, "indicator struct \"%s\" has too many members", ind_name);
     611            0 :         }
     612              : 
     613            0 :         free(pbuf);
     614            0 :         free(ind_pbuf);
     615            0 : }
     616              : 
     617              : void
     618            0 : ECPGfree_struct_member(struct ECPGstruct_member *rm)
     619              : {
     620            0 :         while (rm)
     621              :         {
     622            0 :                 struct ECPGstruct_member *p = rm;
     623              : 
     624            0 :                 rm = rm->next;
     625            0 :                 free(p->name);
     626            0 :                 ECPGfree_type(p->type);
     627            0 :                 free(p);
     628            0 :         }
     629            0 : }
     630              : 
     631              : void
     632            0 : ECPGfree_type(struct ECPGtype *type)
     633              : {
     634            0 :         if (!IS_SIMPLE_TYPE(type->type))
     635              :         {
     636            0 :                 switch (type->type)
     637              :                 {
     638              :                         case ECPGt_array:
     639            0 :                                 switch (type->u.element->type)
     640              :                                 {
     641              :                                         case ECPGt_array:
     642            0 :                                                 base_yyerror("internal error: found multidimensional array\n");
     643            0 :                                                 break;
     644              :                                         case ECPGt_struct:
     645              :                                         case ECPGt_union:
     646              :                                                 /* Array of structs. */
     647            0 :                                                 ECPGfree_type(type->u.element);
     648            0 :                                                 break;
     649              :                                         default:
     650            0 :                                                 if (!IS_SIMPLE_TYPE(type->u.element->type))
     651            0 :                                                         base_yyerror("internal error: unknown datatype, please report this to <" PACKAGE_BUGREPORT ">");
     652              : 
     653            0 :                                                 ECPGfree_type(type->u.element);
     654            0 :                                 }
     655            0 :                                 break;
     656              :                         case ECPGt_struct:
     657              :                         case ECPGt_union:
     658            0 :                                 ECPGfree_struct_member(type->u.members);
     659            0 :                                 break;
     660              :                         default:
     661            0 :                                 mmerror(PARSE_ERROR, ET_ERROR, "unrecognized variable type code %d", type->type);
     662            0 :                                 break;
     663              :                 }
     664            0 :         }
     665            0 :         free(type->type_name);
     666            0 :         free(type->size);
     667            0 :         free(type->struct_sizeof);
     668            0 :         free(type);
     669            0 : }
     670              : 
     671              : const char *
     672            0 : get_dtype(enum ECPGdtype type)
     673              : {
     674            0 :         switch (type)
     675              :         {
     676              :                 case ECPGd_count:
     677            0 :                         return "ECPGd_count";
     678              :                         break;
     679              :                 case ECPGd_data:
     680            0 :                         return "ECPGd_data";
     681              :                         break;
     682              :                 case ECPGd_di_code:
     683            0 :                         return "ECPGd_di_code";
     684              :                         break;
     685              :                 case ECPGd_di_precision:
     686            0 :                         return "ECPGd_di_precision";
     687              :                         break;
     688              :                 case ECPGd_indicator:
     689            0 :                         return "ECPGd_indicator";
     690              :                         break;
     691              :                 case ECPGd_key_member:
     692            0 :                         return "ECPGd_key_member";
     693              :                         break;
     694              :                 case ECPGd_length:
     695            0 :                         return "ECPGd_length";
     696              :                         break;
     697              :                 case ECPGd_name:
     698            0 :                         return "ECPGd_name";
     699              :                         break;
     700              :                 case ECPGd_nullable:
     701            0 :                         return "ECPGd_nullable";
     702              :                         break;
     703              :                 case ECPGd_octet:
     704            0 :                         return "ECPGd_octet";
     705              :                         break;
     706              :                 case ECPGd_precision:
     707            0 :                         return "ECPGd_precision";
     708              :                         break;
     709              :                 case ECPGd_ret_length:
     710            0 :                         return "ECPGd_ret_length";
     711              :                 case ECPGd_ret_octet:
     712            0 :                         return "ECPGd_ret_octet";
     713              :                         break;
     714              :                 case ECPGd_scale:
     715            0 :                         return "ECPGd_scale";
     716              :                         break;
     717              :                 case ECPGd_type:
     718            0 :                         return "ECPGd_type";
     719              :                         break;
     720              :                 case ECPGd_cardinality:
     721            0 :                         return "ECPGd_cardinality";
     722              :                 default:
     723            0 :                         mmerror(PARSE_ERROR, ET_ERROR, "unrecognized descriptor item code %d", type);
     724            0 :         }
     725              : 
     726            0 :         return NULL;
     727            0 : }
        

Generated by: LCOV version 2.3.2-1