LCOV - code coverage report
Current view: top level - src/backend/bootstrap - bootparse.y (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 135 161 83.9 %
Date: 2025-01-18 04:15:08 Functions: 2 2 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : %{
       2             : /*-------------------------------------------------------------------------
       3             :  *
       4             :  * bootparse.y
       5             :  *    yacc grammar for the "bootstrap" mode (BKI file format)
       6             :  *
       7             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       8             :  * Portions Copyright (c) 1994, Regents of the University of California
       9             :  *
      10             :  *
      11             :  * IDENTIFICATION
      12             :  *    src/backend/bootstrap/bootparse.y
      13             :  *
      14             :  *-------------------------------------------------------------------------
      15             :  */
      16             : 
      17             : #include "postgres.h"
      18             : 
      19             : #include <unistd.h>
      20             : 
      21             : #include "bootstrap/bootstrap.h"
      22             : #include "catalog/heap.h"
      23             : #include "catalog/namespace.h"
      24             : #include "catalog/pg_am.h"
      25             : #include "catalog/pg_authid.h"
      26             : #include "catalog/pg_class.h"
      27             : #include "catalog/pg_namespace.h"
      28             : #include "catalog/pg_tablespace.h"
      29             : #include "catalog/toasting.h"
      30             : #include "commands/defrem.h"
      31             : #include "miscadmin.h"
      32             : #include "nodes/makefuncs.h"
      33             : #include "utils/memutils.h"
      34             : 
      35             : #include "bootparse.h"
      36             : 
      37             : 
      38             : /*
      39             :  * Bison doesn't allocate anything that needs to live across parser calls,
      40             :  * so we can easily have it use palloc instead of malloc.  This prevents
      41             :  * memory leaks if we error out during parsing.
      42             :  */
      43             : #define YYMALLOC palloc
      44             : #define YYFREE   pfree
      45             : 
      46             : static MemoryContext per_line_ctx = NULL;
      47             : 
      48             : static void
      49     1000800 : do_start(void)
      50             : {
      51             :     Assert(CurrentMemoryContext == CurTransactionContext);
      52             :     /* First time through, create the per-line working context */
      53     1000800 :     if (per_line_ctx == NULL)
      54          90 :         per_line_ctx = AllocSetContextCreate(CurTransactionContext,
      55             :                                              "bootstrap per-line processing",
      56             :                                              ALLOCSET_DEFAULT_SIZES);
      57     1000800 :     MemoryContextSwitchTo(per_line_ctx);
      58     1000800 : }
      59             : 
      60             : 
      61             : static void
      62     1000800 : do_end(void)
      63             : {
      64             :     /* Reclaim memory allocated while processing this line */
      65     1000800 :     MemoryContextSwitchTo(CurTransactionContext);
      66     1000800 :     MemoryContextReset(per_line_ctx);
      67     1000800 :     CHECK_FOR_INTERRUPTS();     /* allow SIGINT to kill bootstrap run */
      68     1000800 :     if (isatty(0))
      69             :     {
      70           0 :         printf("bootstrap> ");
      71           0 :         fflush(stdout);
      72             :     }
      73     1000800 : }
      74             : 
      75             : 
      76             : static int num_columns_read = 0;
      77             : 
      78             : %}
      79             : 
      80             : %parse-param {yyscan_t yyscanner}
      81             : %lex-param   {yyscan_t yyscanner}
      82             : %pure-parser
      83             : %expect 0
      84             : %name-prefix="boot_yy"
      85             : 
      86             : %union
      87             : {
      88             :     List        *list;
      89             :     IndexElem   *ielem;
      90             :     char        *str;
      91             :     const char  *kw;
      92             :     int         ival;
      93             :     Oid         oidval;
      94             : }
      95             : 
      96             : %type <list>  boot_index_params
      97             : %type <ielem> boot_index_param
      98             : %type <str>   boot_ident
      99             : %type <ival>  optbootstrap optsharedrelation boot_column_nullness
     100             : %type <oidval> oidspec optrowtypeoid
     101             : 
     102             : %token <str> ID
     103             : %token COMMA EQUALS LPAREN RPAREN
     104             : /* NULLVAL is a reserved keyword */
     105             : %token NULLVAL
     106             : /* All the rest are unreserved, and should be handled in boot_ident! */
     107             : %token <kw> OPEN XCLOSE XCREATE INSERT_TUPLE
     108             : %token <kw> XDECLARE INDEX ON USING XBUILD INDICES UNIQUE XTOAST
     109             : %token <kw> OBJ_ID XBOOTSTRAP XSHARED_RELATION XROWTYPE_OID
     110             : %token <kw> XFORCE XNOT XNULL
     111             : 
     112             : %start TopLevel
     113             : 
     114             : %%
     115             : 
     116             : TopLevel:
     117             :           Boot_Queries
     118             :         |
     119             :         ;
     120             : 
     121             : Boot_Queries:
     122             :           Boot_Query
     123             :         | Boot_Queries Boot_Query
     124             :         ;
     125             : 
     126             : Boot_Query :
     127             :           Boot_OpenStmt
     128             :         | Boot_CloseStmt
     129             :         | Boot_CreateStmt
     130             :         | Boot_InsertStmt
     131             :         | Boot_DeclareIndexStmt
     132             :         | Boot_DeclareUniqueIndexStmt
     133             :         | Boot_DeclareToastStmt
     134             :         | Boot_BuildIndsStmt
     135             :         ;
     136             : 
     137             : Boot_OpenStmt:
     138             :           OPEN boot_ident
     139             :                 {
     140        5400 :                     do_start();
     141        5400 :                     boot_openrel($2);
     142        5400 :                     do_end();
     143             : 
     144             :                     (void) yynerrs; /* suppress compiler warning */
     145             :                 }
     146             :         ;
     147             : 
     148             : Boot_CloseStmt:
     149             :           XCLOSE boot_ident
     150             :                 {
     151        5760 :                     do_start();
     152        5760 :                     closerel($2);
     153        5760 :                     do_end();
     154             :                 }
     155             :         ;
     156             : 
     157             : Boot_CreateStmt:
     158             :           XCREATE boot_ident oidspec optbootstrap optsharedrelation optrowtypeoid LPAREN
     159             :                 {
     160        5760 :                     do_start();
     161        5760 :                     numattr = 0;
     162        5760 :                     elog(DEBUG4, "creating%s%s relation %s %u",
     163             :                          $4 ? " bootstrap" : "",
     164             :                          $5 ? " shared" : "",
     165             :                          $2,
     166             :                          $3);
     167             :                 }
     168             :           boot_column_list
     169             :                 {
     170        5760 :                     do_end();
     171             :                 }
     172             :           RPAREN
     173             :                 {
     174             :                     TupleDesc   tupdesc;
     175             :                     bool        shared_relation;
     176             :                     bool        mapped_relation;
     177             : 
     178        5760 :                     do_start();
     179             : 
     180        5760 :                     tupdesc = CreateTupleDesc(numattr, attrtypes);
     181             : 
     182        5760 :                     shared_relation = $5;
     183             : 
     184             :                     /*
     185             :                      * The catalogs that use the relation mapper are the
     186             :                      * bootstrap catalogs plus the shared catalogs.  If this
     187             :                      * ever gets more complicated, we should invent a BKI
     188             :                      * keyword to mark the mapped catalogs, but for now a
     189             :                      * quick hack seems the most appropriate thing.  Note in
     190             :                      * particular that all "nailed" heap rels (see formrdesc
     191             :                      * in relcache.c) must be mapped.
     192             :                      */
     193        5760 :                     mapped_relation = ($4 || shared_relation);
     194             : 
     195        5760 :                     if ($4)
     196             :                     {
     197             :                         TransactionId relfrozenxid;
     198             :                         MultiXactId relminmxid;
     199             : 
     200         360 :                         if (boot_reldesc)
     201             :                         {
     202           0 :                             elog(DEBUG4, "create bootstrap: warning, open relation exists, closing first");
     203           0 :                             closerel(NULL);
     204             :                         }
     205             : 
     206         360 :                         boot_reldesc = heap_create($2,
     207             :                                                    PG_CATALOG_NAMESPACE,
     208             :                                                    shared_relation ? GLOBALTABLESPACE_OID : 0,
     209         360 :                                                    $3,
     210             :                                                    InvalidOid,
     211             :                                                    HEAP_TABLE_AM_OID,
     212             :                                                    tupdesc,
     213             :                                                    RELKIND_RELATION,
     214             :                                                    RELPERSISTENCE_PERMANENT,
     215             :                                                    shared_relation,
     216             :                                                    mapped_relation,
     217             :                                                    true,
     218             :                                                    &relfrozenxid,
     219             :                                                    &relminmxid,
     220             :                                                    true);
     221         360 :                         elog(DEBUG4, "bootstrap relation created");
     222             :                     }
     223             :                     else
     224             :                     {
     225             :                         Oid         id;
     226             : 
     227        5400 :                         id = heap_create_with_catalog($2,
     228             :                                                       PG_CATALOG_NAMESPACE,
     229             :                                                       shared_relation ? GLOBALTABLESPACE_OID : 0,
     230        5400 :                                                       $3,
     231        5400 :                                                       $6,
     232             :                                                       InvalidOid,
     233             :                                                       BOOTSTRAP_SUPERUSERID,
     234             :                                                       HEAP_TABLE_AM_OID,
     235             :                                                       tupdesc,
     236             :                                                       NIL,
     237             :                                                       RELKIND_RELATION,
     238             :                                                       RELPERSISTENCE_PERMANENT,
     239             :                                                       shared_relation,
     240             :                                                       mapped_relation,
     241             :                                                       ONCOMMIT_NOOP,
     242             :                                                       (Datum) 0,
     243             :                                                       false,
     244             :                                                       true,
     245             :                                                       false,
     246             :                                                       InvalidOid,
     247             :                                                       NULL);
     248        5400 :                         elog(DEBUG4, "relation created with OID %u", id);
     249             :                     }
     250        5760 :                     do_end();
     251             :                 }
     252             :         ;
     253             : 
     254             : Boot_InsertStmt:
     255             :           INSERT_TUPLE
     256             :                 {
     257      963630 :                     do_start();
     258      963630 :                     elog(DEBUG4, "inserting row");
     259      963630 :                     num_columns_read = 0;
     260             :                 }
     261             :           LPAREN boot_column_val_list RPAREN
     262             :                 {
     263      963630 :                     if (num_columns_read != numattr)
     264           0 :                         elog(ERROR, "incorrect number of columns in row (expected %d, got %d)",
     265             :                              numattr, num_columns_read);
     266      963630 :                     if (boot_reldesc == NULL)
     267           0 :                         elog(FATAL, "relation not open");
     268      963630 :                     InsertOneTuple();
     269      963630 :                     do_end();
     270             :                 }
     271             :         ;
     272             : 
     273             : Boot_DeclareIndexStmt:
     274             :           XDECLARE INDEX boot_ident oidspec ON boot_ident USING boot_ident LPAREN boot_index_params RPAREN
     275             :                 {
     276        1260 :                     IndexStmt  *stmt = makeNode(IndexStmt);
     277             :                     Oid         relationId;
     278             : 
     279        1260 :                     elog(DEBUG4, "creating index \"%s\"", $3);
     280             : 
     281        1260 :                     do_start();
     282             : 
     283        1260 :                     stmt->idxname = $3;
     284        1260 :                     stmt->relation = makeRangeVar(NULL, $6, -1);
     285        1260 :                     stmt->accessMethod = $8;
     286        1260 :                     stmt->tableSpace = NULL;
     287        1260 :                     stmt->indexParams = $10;
     288        1260 :                     stmt->indexIncludingParams = NIL;
     289        1260 :                     stmt->options = NIL;
     290        1260 :                     stmt->whereClause = NULL;
     291        1260 :                     stmt->excludeOpNames = NIL;
     292        1260 :                     stmt->idxcomment = NULL;
     293        1260 :                     stmt->indexOid = InvalidOid;
     294        1260 :                     stmt->oldNumber = InvalidRelFileNumber;
     295        1260 :                     stmt->oldCreateSubid = InvalidSubTransactionId;
     296        1260 :                     stmt->oldFirstRelfilelocatorSubid = InvalidSubTransactionId;
     297        1260 :                     stmt->unique = false;
     298        1260 :                     stmt->primary = false;
     299        1260 :                     stmt->isconstraint = false;
     300        1260 :                     stmt->deferrable = false;
     301        1260 :                     stmt->initdeferred = false;
     302        1260 :                     stmt->transformed = false;
     303        1260 :                     stmt->concurrent = false;
     304        1260 :                     stmt->if_not_exists = false;
     305        1260 :                     stmt->reset_default_tblspc = false;
     306             : 
     307             :                     /* locks and races need not concern us in bootstrap mode */
     308        1260 :                     relationId = RangeVarGetRelid(stmt->relation, NoLock,
     309             :                                                   false);
     310             : 
     311        1260 :                     DefineIndex(relationId,
     312             :                                 stmt,
     313        1260 :                                 $4,
     314             :                                 InvalidOid,
     315             :                                 InvalidOid,
     316             :                                 -1,
     317             :                                 false,
     318             :                                 false,
     319             :                                 false,
     320             :                                 true, /* skip_build */
     321             :                                 false);
     322        1260 :                     do_end();
     323             :                 }
     324             :         ;
     325             : 
     326             : Boot_DeclareUniqueIndexStmt:
     327             :           XDECLARE UNIQUE INDEX boot_ident oidspec ON boot_ident USING boot_ident LPAREN boot_index_params RPAREN
     328             :                 {
     329        9900 :                     IndexStmt  *stmt = makeNode(IndexStmt);
     330             :                     Oid         relationId;
     331             : 
     332        9900 :                     elog(DEBUG4, "creating unique index \"%s\"", $4);
     333             : 
     334        9900 :                     do_start();
     335             : 
     336        9900 :                     stmt->idxname = $4;
     337        9900 :                     stmt->relation = makeRangeVar(NULL, $7, -1);
     338        9900 :                     stmt->accessMethod = $9;
     339        9900 :                     stmt->tableSpace = NULL;
     340        9900 :                     stmt->indexParams = $11;
     341        9900 :                     stmt->indexIncludingParams = NIL;
     342        9900 :                     stmt->options = NIL;
     343        9900 :                     stmt->whereClause = NULL;
     344        9900 :                     stmt->excludeOpNames = NIL;
     345        9900 :                     stmt->idxcomment = NULL;
     346        9900 :                     stmt->indexOid = InvalidOid;
     347        9900 :                     stmt->oldNumber = InvalidRelFileNumber;
     348        9900 :                     stmt->oldCreateSubid = InvalidSubTransactionId;
     349        9900 :                     stmt->oldFirstRelfilelocatorSubid = InvalidSubTransactionId;
     350        9900 :                     stmt->unique = true;
     351        9900 :                     stmt->primary = false;
     352        9900 :                     stmt->isconstraint = false;
     353        9900 :                     stmt->deferrable = false;
     354        9900 :                     stmt->initdeferred = false;
     355        9900 :                     stmt->transformed = false;
     356        9900 :                     stmt->concurrent = false;
     357        9900 :                     stmt->if_not_exists = false;
     358        9900 :                     stmt->reset_default_tblspc = false;
     359             : 
     360             :                     /* locks and races need not concern us in bootstrap mode */
     361        9900 :                     relationId = RangeVarGetRelid(stmt->relation, NoLock,
     362             :                                                   false);
     363             : 
     364        9900 :                     DefineIndex(relationId,
     365             :                                 stmt,
     366        9900 :                                 $5,
     367             :                                 InvalidOid,
     368             :                                 InvalidOid,
     369             :                                 -1,
     370             :                                 false,
     371             :                                 false,
     372             :                                 false,
     373             :                                 true, /* skip_build */
     374             :                                 false);
     375        9900 :                     do_end();
     376             :                 }
     377             :         ;
     378             : 
     379             : Boot_DeclareToastStmt:
     380             :           XDECLARE XTOAST oidspec oidspec ON boot_ident
     381             :                 {
     382        3240 :                     elog(DEBUG4, "creating toast table for table \"%s\"", $6);
     383             : 
     384        3240 :                     do_start();
     385             : 
     386        3240 :                     BootstrapToastTable($6, $3, $4);
     387        3240 :                     do_end();
     388             :                 }
     389             :         ;
     390             : 
     391             : Boot_BuildIndsStmt:
     392             :           XBUILD INDICES
     393             :                 {
     394          90 :                     do_start();
     395          90 :                     build_indices();
     396          90 :                     do_end();
     397             :                 }
     398             :         ;
     399             : 
     400             : 
     401             : boot_index_params:
     402        7290 :         boot_index_params COMMA boot_index_param    { $$ = lappend($1, $3); }
     403       11160 :         | boot_index_param                          { $$ = list_make1($1); }
     404             :         ;
     405             : 
     406             : boot_index_param:
     407             :         boot_ident boot_ident
     408             :                 {
     409       18450 :                     IndexElem  *n = makeNode(IndexElem);
     410             : 
     411       18450 :                     n->name = $1;
     412       18450 :                     n->expr = NULL;
     413       18450 :                     n->indexcolname = NULL;
     414       18450 :                     n->collation = NIL;
     415       18450 :                     n->opclass = list_make1(makeString($2));
     416       18450 :                     n->ordering = SORTBY_DEFAULT;
     417       18450 :                     n->nulls_ordering = SORTBY_NULLS_DEFAULT;
     418       18450 :                     $$ = n;
     419             :                 }
     420             :         ;
     421             : 
     422             : optbootstrap:
     423         360 :             XBOOTSTRAP  { $$ = 1; }
     424        5400 :         |               { $$ = 0; }
     425             :         ;
     426             : 
     427             : optsharedrelation:
     428         990 :             XSHARED_RELATION    { $$ = 1; }
     429        4770 :         |                       { $$ = 0; }
     430             :         ;
     431             : 
     432             : optrowtypeoid:
     433         810 :             XROWTYPE_OID oidspec    { $$ = $2; }
     434        4950 :         |                           { $$ = InvalidOid; }
     435             :         ;
     436             : 
     437             : boot_column_list:
     438             :           boot_column_def
     439             :         | boot_column_list COMMA boot_column_def
     440             :         ;
     441             : 
     442             : boot_column_def:
     443             :           boot_ident EQUALS boot_ident boot_column_nullness
     444             :                 {
     445       54000 :                    if (++numattr > MAXATTR)
     446           0 :                         elog(FATAL, "too many columns");
     447       54000 :                    DefineAttr($1, $3, numattr-1, $4);
     448             :                 }
     449             :         ;
     450             : 
     451             : boot_column_nullness:
     452        3060 :             XFORCE XNOT XNULL   { $$ = BOOTCOL_NULL_FORCE_NOT_NULL; }
     453         360 :         |   XFORCE XNULL        {  $$ = BOOTCOL_NULL_FORCE_NULL; }
     454       50580 :         | { $$ = BOOTCOL_NULL_AUTO; }
     455             :         ;
     456             : 
     457             : oidspec:
     458       24210 :             boot_ident                          { $$ = atooid($1); }
     459             :         ;
     460             : 
     461             : boot_column_val_list:
     462             :            boot_column_val
     463             :         |  boot_column_val_list boot_column_val
     464             :         |  boot_column_val_list COMMA boot_column_val
     465             :         ;
     466             : 
     467             : boot_column_val:
     468             :           boot_ident
     469    11475468 :             { InsertOneValue($1, num_columns_read++); }
     470             :         | NULLVAL
     471     2825892 :             { InsertOneNull(num_columns_read++); }
     472             :         ;
     473             : 
     474             : boot_ident:
     475    11698218 :           ID            { $$ = $1; }
     476           0 :         | OPEN          { $$ = pstrdup($1); }
     477           0 :         | XCLOSE        { $$ = pstrdup($1); }
     478           0 :         | XCREATE       { $$ = pstrdup($1); }
     479           0 :         | INSERT_TUPLE  { $$ = pstrdup($1); }
     480           0 :         | XDECLARE      { $$ = pstrdup($1); }
     481           0 :         | INDEX         { $$ = pstrdup($1); }
     482           0 :         | ON            { $$ = pstrdup($1); }
     483           0 :         | USING         { $$ = pstrdup($1); }
     484           0 :         | XBUILD        { $$ = pstrdup($1); }
     485           0 :         | INDICES       { $$ = pstrdup($1); }
     486           0 :         | UNIQUE        { $$ = pstrdup($1); }
     487           0 :         | XTOAST        { $$ = pstrdup($1); }
     488           0 :         | OBJ_ID        { $$ = pstrdup($1); }
     489           0 :         | XBOOTSTRAP    { $$ = pstrdup($1); }
     490           0 :         | XSHARED_RELATION  { $$ = pstrdup($1); }
     491           0 :         | XROWTYPE_OID  { $$ = pstrdup($1); }
     492           0 :         | XFORCE        { $$ = pstrdup($1); }
     493           0 :         | XNOT          { $$ = pstrdup($1); }
     494           0 :         | XNULL         { $$ = pstrdup($1); }
     495             :         ;
     496             : %%

Generated by: LCOV version 1.14