LCOV - code coverage report
Current view: top level - src/interfaces/ecpg/preproc - ecpg.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 138 252 54.8 %
Date: 2025-01-18 05:15:39 Functions: 3 4 75.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* src/interfaces/ecpg/preproc/ecpg.c */
       2             : 
       3             : /* Main for ecpg, the PostgreSQL embedded SQL precompiler. */
       4             : /* Copyright (c) 1996-2025, PostgreSQL Global Development Group */
       5             : 
       6             : #include "postgres_fe.h"
       7             : 
       8             : #include <unistd.h>
       9             : 
      10             : #include "getopt_long.h"
      11             : 
      12             : #include "preproc_extern.h"
      13             : 
      14             : int         ret_value = 0;
      15             : bool        autocommit = false,
      16             :             auto_create_c = false,
      17             :             system_includes = false,
      18             :             force_indicator = true,
      19             :             questionmarks = false,
      20             :             regression_mode = false,
      21             :             auto_prepare = false;
      22             : 
      23             : static const char *progname;
      24             : char       *output_filename;
      25             : 
      26             : enum COMPAT_MODE compat = ECPG_COMPAT_PGSQL;
      27             : 
      28             : struct _include_path *include_paths = NULL;
      29             : struct cursor *cur = NULL;
      30             : struct typedefs *types = NULL;
      31             : struct _defines *defines = NULL;
      32             : struct declared_list *g_declared_list = NULL;
      33             : 
      34             : static void
      35           0 : help(const char *progname)
      36             : {
      37           0 :     printf(_("%s is the PostgreSQL embedded SQL preprocessor for C programs.\n\n"),
      38             :            progname);
      39           0 :     printf(_("Usage:\n"
      40             :              "  %s [OPTION]... FILE...\n\n"),
      41             :            progname);
      42           0 :     printf(_("Options:\n"));
      43           0 :     printf(_("  -c             automatically generate C code from embedded SQL code;\n"
      44             :              "                 this affects EXEC SQL TYPE\n"));
      45           0 :     printf(_("  -C MODE        set compatibility mode; MODE can be one of\n"
      46             :              "                 \"INFORMIX\", \"INFORMIX_SE\", \"ORACLE\"\n"));
      47             : #ifdef YYDEBUG
      48             :     printf(_("  -d             generate parser debug output\n"));
      49             : #endif
      50           0 :     printf(_("  -D SYMBOL      define SYMBOL\n"));
      51           0 :     printf(_("  -h             parse a header file, this option includes option \"-c\"\n"));
      52           0 :     printf(_("  -i             parse system include files as well\n"));
      53           0 :     printf(_("  -I DIRECTORY   search DIRECTORY for include files\n"));
      54           0 :     printf(_("  -o OUTFILE     write result to OUTFILE\n"));
      55           0 :     printf(_("  -r OPTION      specify run-time behavior; OPTION can be:\n"
      56             :              "                 \"no_indicator\", \"prepare\", \"questionmarks\"\n"));
      57           0 :     printf(_("  --regression   run in regression testing mode\n"));
      58           0 :     printf(_("  -t             turn on autocommit of transactions\n"));
      59           0 :     printf(_("  -V, --version  output version information, then exit\n"));
      60           0 :     printf(_("  -?, --help     show this help, then exit\n"));
      61           0 :     printf(_("\nIf no output file is specified, the name is formed by adding .c to the\n"
      62             :              "input file name, after stripping off .pgc if present.\n"));
      63           0 :     printf(_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
      64           0 :     printf(_("%s home page: <%s>\n"), PACKAGE_NAME, PACKAGE_URL);
      65           0 : }
      66             : 
      67             : static void
      68         812 : add_include_path(char *path)
      69             : {
      70         812 :     struct _include_path *ip = include_paths,
      71             :                *new;
      72             : 
      73         812 :     new = mm_alloc(sizeof(struct _include_path));
      74         812 :     new->path = path;
      75         812 :     new->next = NULL;
      76             : 
      77         812 :     if (ip == NULL)
      78         132 :         include_paths = new;
      79             :     else
      80             :     {
      81        2100 :         for (; ip->next != NULL; ip = ip->next);
      82         680 :         ip->next = new;
      83             :     }
      84         812 : }
      85             : 
      86             : /*
      87             :  * Process a command line -D switch
      88             :  */
      89             : static void
      90           2 : add_preprocessor_define(char *define)
      91             : {
      92             :     /* copy the argument to avoid relying on argv storage */
      93           2 :     char       *define_copy = mm_strdup(define);
      94             :     char       *ptr;
      95             :     struct _defines *newdef;
      96             : 
      97           2 :     newdef = mm_alloc(sizeof(struct _defines));
      98             : 
      99             :     /* look for = sign */
     100           2 :     ptr = strchr(define_copy, '=');
     101           2 :     if (ptr != NULL)
     102             :     {
     103             :         /* symbol has a value */
     104             :         char       *tmp;
     105             : 
     106             :         /* strip any spaces between name and '=' */
     107           2 :         for (tmp = ptr - 1; tmp >= define_copy && *tmp == ' '; tmp--);
     108           2 :         tmp[1] = '\0';
     109             : 
     110             :         /*
     111             :          * Note we don't bother to separately malloc cmdvalue; it will never
     112             :          * be freed so that's not necessary.
     113             :          */
     114           2 :         newdef->cmdvalue = ptr + 1;
     115             :     }
     116             :     else
     117             :     {
     118             :         /* define it as "1"; again no need to malloc it */
     119           0 :         newdef->cmdvalue = "1";
     120             :     }
     121           2 :     newdef->name = define_copy;
     122           2 :     newdef->value = mm_strdup(newdef->cmdvalue);
     123           2 :     newdef->used = NULL;
     124           2 :     newdef->next = defines;
     125           2 :     defines = newdef;
     126           2 : }
     127             : 
     128             : #define ECPG_GETOPT_LONG_REGRESSION     1
     129             : int
     130         132 : main(int argc, char *const argv[])
     131             : {
     132             :     static struct option ecpg_options[] = {
     133             :         {"regression", no_argument, NULL, ECPG_GETOPT_LONG_REGRESSION},
     134             :         {NULL, 0, NULL, 0}
     135             :     };
     136             : 
     137             :     int         fnr,
     138             :                 c,
     139         132 :                 out_option = 0;
     140         132 :     bool        verbose = false,
     141         132 :                 header_mode = false;
     142             :     struct _include_path *ip;
     143             :     char        my_exec_path[MAXPGPATH];
     144             :     char        include_path[MAXPGPATH];
     145             : 
     146         132 :     set_pglocale_pgservice(argv[0], PG_TEXTDOMAIN("ecpg"));
     147             : 
     148         132 :     progname = get_progname(argv[0]);
     149             : 
     150         132 :     if (find_my_exec(argv[0], my_exec_path) < 0)
     151             :     {
     152           0 :         fprintf(stderr, _("%s: could not locate my own executable path\n"), argv[0]);
     153           0 :         return ILLEGAL_OPTION;
     154             :     }
     155             : 
     156         132 :     if (argc > 1)
     157             :     {
     158         132 :         if (strcmp(argv[1], "--help") == 0 || strcmp(argv[1], "-?") == 0)
     159             :         {
     160           0 :             help(progname);
     161           0 :             exit(0);
     162             :         }
     163         132 :         if (strcmp(argv[1], "--version") == 0 || strcmp(argv[1], "-V") == 0)
     164             :         {
     165           0 :             printf("ecpg (PostgreSQL) %s\n", PG_VERSION);
     166           0 :             exit(0);
     167             :         }
     168             :     }
     169             : 
     170         132 :     output_filename = NULL;
     171         696 :     while ((c = getopt_long(argc, argv, "cC:dD:hiI:o:r:tv", ecpg_options, NULL)) != -1)
     172             :     {
     173         564 :         switch (c)
     174             :         {
     175           4 :             case 'c':
     176           4 :                 auto_create_c = true;
     177           4 :                 break;
     178          22 :             case 'C':
     179          22 :                 if (pg_strcasecmp(optarg, "INFORMIX") == 0 || pg_strcasecmp(optarg, "INFORMIX_SE") == 0)
     180          20 :                 {
     181             :                     char        pkginclude_path[MAXPGPATH];
     182             :                     char        informix_path[MAXPGPATH];
     183             : 
     184          20 :                     compat = (pg_strcasecmp(optarg, "INFORMIX") == 0) ? ECPG_COMPAT_INFORMIX : ECPG_COMPAT_INFORMIX_SE;
     185          20 :                     get_pkginclude_path(my_exec_path, pkginclude_path);
     186          20 :                     snprintf(informix_path, MAXPGPATH, "%s/informix/esql", pkginclude_path);
     187          20 :                     add_include_path(informix_path);
     188             :                 }
     189           2 :                 else if (pg_strcasecmp(optarg, "ORACLE") == 0)
     190             :                 {
     191           2 :                     compat = ECPG_COMPAT_ORACLE;
     192             :                 }
     193             :                 else
     194             :                 {
     195           0 :                     fprintf(stderr, _("Try \"%s --help\" for more information.\n"), argv[0]);
     196           0 :                     return ILLEGAL_OPTION;
     197             :                 }
     198          22 :                 break;
     199           0 :             case 'd':
     200             : #ifdef YYDEBUG
     201             :                 base_yydebug = 1;
     202             : #else
     203           0 :                 fprintf(stderr, _("%s: parser debug support (-d) not available\n"),
     204             :                         progname);
     205             : #endif
     206           0 :                 break;
     207           2 :             case 'D':
     208           2 :                 add_preprocessor_define(optarg);
     209           2 :                 break;
     210           0 :             case 'h':
     211           0 :                 header_mode = true;
     212             :                 /* this must include "-c" to make sense: */
     213           0 :                 auto_create_c = true;
     214           0 :                 break;
     215           2 :             case 'i':
     216           2 :                 system_includes = true;
     217           2 :                 break;
     218         264 :             case 'I':
     219         264 :                 add_include_path(optarg);
     220         264 :                 break;
     221         132 :             case 'o':
     222         132 :                 output_filename = mm_strdup(optarg);
     223         132 :                 if (strcmp(output_filename, "-") == 0)
     224           0 :                     base_yyout = stdout;
     225             :                 else
     226         132 :                     base_yyout = fopen(output_filename, PG_BINARY_W);
     227             : 
     228         132 :                 if (base_yyout == NULL)
     229             :                 {
     230           0 :                     fprintf(stderr, _("%s: could not open file \"%s\": %m\n"),
     231             :                             progname, output_filename);
     232           0 :                     output_filename = NULL;
     233             :                 }
     234             :                 else
     235         132 :                     out_option = 1;
     236         132 :                 break;
     237           6 :             case 'r':
     238           6 :                 if (pg_strcasecmp(optarg, "no_indicator") == 0)
     239           2 :                     force_indicator = false;
     240           4 :                 else if (pg_strcasecmp(optarg, "prepare") == 0)
     241           2 :                     auto_prepare = true;
     242           2 :                 else if (pg_strcasecmp(optarg, "questionmarks") == 0)
     243           2 :                     questionmarks = true;
     244             :                 else
     245             :                 {
     246           0 :                     fprintf(stderr, _("Try \"%s --help\" for more information.\n"), argv[0]);
     247           0 :                     return ILLEGAL_OPTION;
     248             :                 }
     249           6 :                 break;
     250           0 :             case 't':
     251           0 :                 autocommit = true;
     252           0 :                 break;
     253           0 :             case 'v':
     254           0 :                 verbose = true;
     255           0 :                 break;
     256         132 :             case ECPG_GETOPT_LONG_REGRESSION:
     257         132 :                 regression_mode = true;
     258         132 :                 break;
     259           0 :             default:
     260           0 :                 fprintf(stderr, _("Try \"%s --help\" for more information.\n"), argv[0]);
     261           0 :                 return ILLEGAL_OPTION;
     262             :         }
     263             :     }
     264             : 
     265         132 :     add_include_path(".");
     266         132 :     add_include_path("/usr/local/include");
     267         132 :     get_include_path(my_exec_path, include_path);
     268         132 :     add_include_path(include_path);
     269         132 :     add_include_path("/usr/include");
     270             : 
     271         132 :     if (verbose)
     272             :     {
     273           0 :         fprintf(stderr,
     274           0 :                 _("%s, the PostgreSQL embedded C preprocessor, version %s\n"),
     275             :                 progname, PG_VERSION);
     276           0 :         fprintf(stderr, _("EXEC SQL INCLUDE ... search starts here:\n"));
     277           0 :         for (ip = include_paths; ip != NULL; ip = ip->next)
     278           0 :             fprintf(stderr, " %s\n", ip->path);
     279           0 :         fprintf(stderr, _("end of search list\n"));
     280           0 :         return 0;
     281             :     }
     282             : 
     283         132 :     if (optind >= argc)          /* no files specified */
     284             :     {
     285           0 :         fprintf(stderr, _("%s: no input files specified\n"), progname);
     286           0 :         fprintf(stderr, _("Try \"%s --help\" for more information.\n"), argv[0]);
     287           0 :         return ILLEGAL_OPTION;
     288             :     }
     289             :     else
     290             :     {
     291             :         /* after the options there must not be anything but filenames */
     292         266 :         for (fnr = optind; fnr < argc; fnr++)
     293             :         {
     294             :             char       *ptr2ext;
     295             : 
     296             :             /* If argv[fnr] is "-" we have to read from stdin */
     297         134 :             if (strcmp(argv[fnr], "-") == 0)
     298             :             {
     299           0 :                 input_filename = mm_alloc(strlen("stdin") + 1);
     300           0 :                 strcpy(input_filename, "stdin");
     301           0 :                 base_yyin = stdin;
     302             :             }
     303             :             else
     304             :             {
     305         134 :                 input_filename = mm_alloc(strlen(argv[fnr]) + 5);
     306         134 :                 strcpy(input_filename, argv[fnr]);
     307             : 
     308             :                 /* take care of relative paths */
     309         134 :                 ptr2ext = last_dir_separator(input_filename);
     310         134 :                 ptr2ext = (ptr2ext ? strrchr(ptr2ext, '.') : strrchr(input_filename, '.'));
     311             : 
     312             :                 /* no extension? */
     313         134 :                 if (ptr2ext == NULL)
     314             :                 {
     315           0 :                     ptr2ext = input_filename + strlen(input_filename);
     316             : 
     317             :                     /* no extension => add .pgc or .pgh */
     318           0 :                     ptr2ext[0] = '.';
     319           0 :                     ptr2ext[1] = 'p';
     320           0 :                     ptr2ext[2] = 'g';
     321           0 :                     ptr2ext[3] = (header_mode == true) ? 'h' : 'c';
     322           0 :                     ptr2ext[4] = '\0';
     323             :                 }
     324             : 
     325         134 :                 base_yyin = fopen(input_filename, PG_BINARY_R);
     326             :             }
     327             : 
     328         134 :             if (out_option == 0)    /* calculate the output name */
     329             :             {
     330           0 :                 if (strcmp(input_filename, "stdin") == 0)
     331           0 :                     base_yyout = stdout;
     332             :                 else
     333             :                 {
     334           0 :                     output_filename = mm_alloc(strlen(input_filename) + 3);
     335           0 :                     strcpy(output_filename, input_filename);
     336             : 
     337           0 :                     ptr2ext = strrchr(output_filename, '.');
     338             :                     /* make extension = .c resp. .h */
     339           0 :                     ptr2ext[1] = (header_mode == true) ? 'h' : 'c';
     340           0 :                     ptr2ext[2] = '\0';
     341             : 
     342           0 :                     base_yyout = fopen(output_filename, PG_BINARY_W);
     343           0 :                     if (base_yyout == NULL)
     344             :                     {
     345           0 :                         fprintf(stderr, _("%s: could not open file \"%s\": %m\n"),
     346             :                                 progname, output_filename);
     347           0 :                         free(output_filename);
     348           0 :                         output_filename = NULL;
     349           0 :                         free(input_filename);
     350           0 :                         continue;
     351             :                     }
     352             :                 }
     353             :             }
     354             : 
     355         134 :             if (base_yyin == NULL)
     356           0 :                 fprintf(stderr, _("%s: could not open file \"%s\": %m\n"),
     357           0 :                         progname, argv[fnr]);
     358             :             else
     359             :             {
     360             :                 struct cursor *ptr;
     361             :                 struct _defines *defptr;
     362             :                 struct _defines *prevdefptr;
     363             :                 struct _defines *nextdefptr;
     364             :                 struct typedefs *typeptr;
     365             :                 struct declared_list *list;
     366             : 
     367             :                 /* remove old cursor definitions if any are still there */
     368         134 :                 for (ptr = cur; ptr != NULL;)
     369             :                 {
     370           0 :                     struct cursor *this = ptr;
     371             :                     struct arguments *l1,
     372             :                                *l2;
     373             : 
     374           0 :                     free(ptr->command);
     375           0 :                     free(ptr->connection);
     376           0 :                     free(ptr->name);
     377           0 :                     for (l1 = ptr->argsinsert; l1; l1 = l2)
     378             :                     {
     379           0 :                         l2 = l1->next;
     380           0 :                         free(l1);
     381             :                     }
     382           0 :                     for (l1 = ptr->argsresult; l1; l1 = l2)
     383             :                     {
     384           0 :                         l2 = l1->next;
     385           0 :                         free(l1);
     386             :                     }
     387           0 :                     ptr = ptr->next;
     388           0 :                     free(this);
     389             :                 }
     390         134 :                 cur = NULL;
     391             : 
     392             :                 /* remove old declared statements if any are still there */
     393         134 :                 for (list = g_declared_list; list != NULL;)
     394             :                 {
     395           0 :                     struct declared_list *this = list;
     396             : 
     397           0 :                     list = list->next;
     398           0 :                     free(this);
     399             :                 }
     400             : 
     401             :                 /* restore defines to their command-line state */
     402         134 :                 prevdefptr = NULL;
     403         140 :                 for (defptr = defines; defptr != NULL; defptr = nextdefptr)
     404             :                 {
     405           6 :                     nextdefptr = defptr->next;
     406           6 :                     if (defptr->cmdvalue != NULL)
     407             :                     {
     408             :                         /* keep it, resetting the value */
     409           4 :                         free(defptr->value);
     410           4 :                         defptr->value = mm_strdup(defptr->cmdvalue);
     411           4 :                         prevdefptr = defptr;
     412             :                     }
     413             :                     else
     414             :                     {
     415             :                         /* remove it */
     416           2 :                         if (prevdefptr != NULL)
     417           0 :                             prevdefptr->next = nextdefptr;
     418             :                         else
     419           2 :                             defines = nextdefptr;
     420           2 :                         free(defptr->name);
     421           2 :                         free(defptr->value);
     422           2 :                         free(defptr);
     423             :                     }
     424             :                 }
     425             : 
     426             :                 /* and old typedefs */
     427         134 :                 for (typeptr = types; typeptr != NULL;)
     428             :                 {
     429           0 :                     struct typedefs *this = typeptr;
     430             : 
     431           0 :                     free(typeptr->name);
     432           0 :                     ECPGfree_struct_member(typeptr->struct_member_list);
     433           0 :                     free(typeptr->type);
     434           0 :                     typeptr = typeptr->next;
     435           0 :                     free(this);
     436             :                 }
     437         134 :                 types = NULL;
     438             : 
     439             :                 /* initialize whenever structures */
     440         134 :                 memset(&when_error, 0, sizeof(struct when));
     441         134 :                 memset(&when_nf, 0, sizeof(struct when));
     442         134 :                 memset(&when_warn, 0, sizeof(struct when));
     443             : 
     444             :                 /* and structure member lists */
     445         134 :                 memset(struct_member_list, 0, sizeof(struct_member_list));
     446             : 
     447             :                 /*
     448             :                  * and our variable counter for out of scope cursors'
     449             :                  * variables
     450             :                  */
     451         134 :                 ecpg_internal_var = 0;
     452             : 
     453             :                 /* finally the actual connection */
     454         134 :                 connection = NULL;
     455             : 
     456             :                 /* initialize lex */
     457         134 :                 lex_init();
     458             : 
     459             :                 /* we need several includes */
     460             :                 /* but not if we are in header mode */
     461         134 :                 if (regression_mode)
     462         134 :                     fprintf(base_yyout, "/* Processed by ecpg (regression mode) */\n");
     463             :                 else
     464           0 :                     fprintf(base_yyout, "/* Processed by ecpg (%s) */\n", PG_VERSION);
     465             : 
     466         134 :                 if (header_mode == false)
     467             :                 {
     468         134 :                     fprintf(base_yyout, "/* These include files are added by the preprocessor */\n#include <ecpglib.h>\n#include <ecpgerrno.h>\n#include <sqlca.h>\n");
     469             : 
     470             :                     /* add some compatibility headers */
     471         134 :                     if (INFORMIX_MODE)
     472          20 :                         fprintf(base_yyout, "/* Needed for informix compatibility */\n#include <ecpg_informix.h>\n");
     473             : 
     474         134 :                     fprintf(base_yyout, "/* End of automatic include section */\n");
     475             :                 }
     476             : 
     477         134 :                 if (regression_mode)
     478         134 :                     fprintf(base_yyout, "#define ECPGdebug(X,Y) ECPGdebug((X)+100,(Y))\n");
     479             : 
     480         134 :                 output_line_number();
     481             : 
     482             :                 /* and parse the source */
     483         134 :                 base_yyparse();
     484             : 
     485             :                 /*
     486             :                  * Check whether all cursors were indeed opened.  It does not
     487             :                  * really make sense to declare a cursor but not open it.
     488             :                  */
     489         208 :                 for (ptr = cur; ptr != NULL; ptr = ptr->next)
     490          74 :                     if (!(ptr->opened))
     491           0 :                         mmerror(PARSE_ERROR, ET_WARNING, "cursor \"%s\" has been declared but not opened", ptr->name);
     492             : 
     493         134 :                 if (base_yyin != NULL && base_yyin != stdin)
     494         134 :                     fclose(base_yyin);
     495         134 :                 if (out_option == 0 && base_yyout != stdout)
     496           0 :                     fclose(base_yyout);
     497             : 
     498             :                 /*
     499             :                  * If there was an error, delete the output file.
     500             :                  */
     501         134 :                 if (ret_value != 0)
     502             :                 {
     503           0 :                     if (strcmp(output_filename, "-") != 0 && unlink(output_filename) != 0)
     504           0 :                         fprintf(stderr, _("could not remove output file \"%s\"\n"), output_filename);
     505             :                 }
     506             :             }
     507             : 
     508         134 :             if (output_filename && out_option == 0)
     509             :             {
     510           0 :                 free(output_filename);
     511           0 :                 output_filename = NULL;
     512             :             }
     513             : 
     514         134 :             free(input_filename);
     515             :         }
     516             :     }
     517         132 :     return ret_value;
     518             : }

Generated by: LCOV version 1.14