PostgreSQL Source Code git master
Loading...
Searching...
No Matches
bootstrap.c
Go to the documentation of this file.
1/*-------------------------------------------------------------------------
2 *
3 * bootstrap.c
4 * routines to support running postgres in 'bootstrap' mode
5 * bootstrap mode is used to create the initial template database
6 *
7 * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
8 * Portions Copyright (c) 1994, Regents of the University of California
9 *
10 * IDENTIFICATION
11 * src/backend/bootstrap/bootstrap.c
12 *
13 *-------------------------------------------------------------------------
14 */
15#include "postgres.h"
16
17#include <unistd.h>
18#include <signal.h>
19
20#include "access/genam.h"
21#include "access/heapam.h"
22#include "access/htup_details.h"
23#include "access/tableam.h"
25#include "access/xact.h"
26#include "bootstrap/bootstrap.h"
27#include "catalog/index.h"
28#include "catalog/pg_authid.h"
30#include "catalog/pg_proc.h"
31#include "catalog/pg_type.h"
32#include "common/link-canary.h"
33#include "miscadmin.h"
34#include "nodes/makefuncs.h"
35#include "pg_getopt.h"
37#include "storage/bufpage.h"
38#include "storage/fd.h"
39#include "storage/ipc.h"
40#include "storage/proc.h"
41#include "utils/builtins.h"
42#include "utils/fmgroids.h"
43#include "utils/guc.h"
44#include "utils/memutils.h"
45#include "utils/rel.h"
46#include "utils/relmapper.h"
47
48
49static void CheckerModeMain(void);
50static void bootstrap_signals(void);
52static void InsertOneProargdefaultsValue(char *value);
53static void populate_typ_list(void);
54static Oid gettype(char *type);
55static void cleanup(void);
56
57/* ----------------
58 * global variables
59 * ----------------
60 */
61
62Relation boot_reldesc; /* current relation descriptor */
63
64Form_pg_attribute attrtypes[MAXATTR]; /* points to attribute info */
65int numattr; /* number of attributes for cur. rel */
66
67
68/*
69 * Basic information associated with each type. This is used before
70 * pg_type is filled, so it has to cover the datatypes used as column types
71 * in the core "bootstrapped" catalogs.
72 *
73 * XXX several of these input/output functions do catalog scans
74 * (e.g., F_REGPROCIN scans pg_proc). this obviously creates some
75 * order dependencies in the catalog creation process.
76 */
90
91static const struct typinfo TypInfo[] = {
92 {"bool", BOOLOID, 0, 1, true, TYPALIGN_CHAR, TYPSTORAGE_PLAIN, InvalidOid,
94 {"bytea", BYTEAOID, 0, -1, false, TYPALIGN_INT, TYPSTORAGE_EXTENDED, InvalidOid,
96 {"char", CHAROID, 0, 1, true, TYPALIGN_CHAR, TYPSTORAGE_PLAIN, InvalidOid,
98 {"cstring", CSTRINGOID, 0, -2, false, TYPALIGN_CHAR, TYPSTORAGE_PLAIN, InvalidOid,
100 {"int2", INT2OID, 0, 2, true, TYPALIGN_SHORT, TYPSTORAGE_PLAIN, InvalidOid,
102 {"int4", INT4OID, 0, 4, true, TYPALIGN_INT, TYPSTORAGE_PLAIN, InvalidOid,
104 {"int8", INT8OID, 0, 8, true, TYPALIGN_DOUBLE, TYPSTORAGE_PLAIN, InvalidOid,
106 {"float4", FLOAT4OID, 0, 4, true, TYPALIGN_INT, TYPSTORAGE_PLAIN, InvalidOid,
108 {"float8", FLOAT8OID, 0, 8, true, TYPALIGN_DOUBLE, TYPSTORAGE_PLAIN, InvalidOid,
112 {"regproc", REGPROCOID, 0, 4, true, TYPALIGN_INT, TYPSTORAGE_PLAIN, InvalidOid,
116 {"jsonb", JSONBOID, 0, -1, false, TYPALIGN_INT, TYPSTORAGE_EXTENDED, InvalidOid,
118 {"oid", OIDOID, 0, 4, true, TYPALIGN_INT, TYPSTORAGE_PLAIN, InvalidOid,
120 {"aclitem", ACLITEMOID, 0, 16, false, TYPALIGN_DOUBLE, TYPSTORAGE_PLAIN, InvalidOid,
124 {"int2vector", INT2VECTOROID, INT2OID, -1, false, TYPALIGN_INT, TYPSTORAGE_PLAIN, InvalidOid,
126 {"oidvector", OIDVECTOROID, OIDOID, -1, false, TYPALIGN_INT, TYPSTORAGE_PLAIN, InvalidOid,
138};
139
140static const int n_types = sizeof(TypInfo) / sizeof(struct typinfo);
141
142struct typmap
143{ /* a hack */
146};
147
148static List *Typ = NIL; /* List of struct typmap* */
149static struct typmap *Ap = NULL;
150
151/*
152 * Basic information about built-in roles.
153 *
154 * Presently, this need only list roles that are mentioned in aclitem arrays
155 * in the catalog .dat files. We might as well list everything that is in
156 * pg_authid.dat, since there aren't that many. Like pg_authid.dat, we
157 * represent the bootstrap superuser's name as "POSTGRES", even though it
158 * (probably) won't be that in the finished installation; this means aclitem
159 * entries in .dat files must spell it like that.
160 */
162{
163 const char *rolname;
165};
166
167static const struct rolinfo RolInfo[] = {
168 {"POSTGRES", BOOTSTRAP_SUPERUSERID},
169 {"pg_database_owner", ROLE_PG_DATABASE_OWNER},
170 {"pg_read_all_data", ROLE_PG_READ_ALL_DATA},
171 {"pg_write_all_data", ROLE_PG_WRITE_ALL_DATA},
172 {"pg_monitor", ROLE_PG_MONITOR},
173 {"pg_read_all_settings", ROLE_PG_READ_ALL_SETTINGS},
174 {"pg_read_all_stats", ROLE_PG_READ_ALL_STATS},
175 {"pg_stat_scan_tables", ROLE_PG_STAT_SCAN_TABLES},
176 {"pg_read_server_files", ROLE_PG_READ_SERVER_FILES},
177 {"pg_write_server_files", ROLE_PG_WRITE_SERVER_FILES},
178 {"pg_execute_server_program", ROLE_PG_EXECUTE_SERVER_PROGRAM},
179 {"pg_signal_backend", ROLE_PG_SIGNAL_BACKEND},
180 {"pg_checkpoint", ROLE_PG_CHECKPOINT},
181 {"pg_maintain", ROLE_PG_MAINTAIN},
182 {"pg_use_reserved_connections", ROLE_PG_USE_RESERVED_CONNECTIONS},
183 {"pg_create_subscription", ROLE_PG_CREATE_SUBSCRIPTION},
184 {"pg_signal_autovacuum_worker", ROLE_PG_SIGNAL_AUTOVACUUM_WORKER}
185};
186
187
188static Datum values[MAXATTR]; /* current row's attribute values */
189static bool Nulls[MAXATTR];
190
191static MemoryContext nogc = NULL; /* special no-gc mem context */
192
193/*
194 * At bootstrap time, we first declare all the indices to be built, and
195 * then build them. The IndexList structure stores enough information
196 * to allow us to build the indices after they've been declared.
197 */
198
206
208
209
210/*
211 * In shared memory checker mode, all we really want to do is create shared
212 * memory and semaphores (just to prove we can do it with the current GUC
213 * settings). Since, in fact, that was already done by
214 * CreateSharedMemoryAndSemaphores(), we have nothing more to do here.
215 */
216static void
218{
219 proc_exit(0);
220}
221
222/*
223 * The main entry point for running the backend in bootstrap mode
224 *
225 * The bootstrap mode is used to initialize the template database.
226 * The bootstrap backend doesn't speak SQL, but instead expects
227 * commands in a special bootstrap language.
228 *
229 * When check_only is true, startup is done only far enough to verify that
230 * the current configuration, particularly the passed in options pertaining
231 * to shared memory sizing, options work (or at least do not cause an error
232 * up to shared memory creation).
233 */
234void
235BootstrapModeMain(int argc, char *argv[], bool check_only)
236{
237 int i;
238 char *progname = argv[0];
239 int flag;
240 char *userDoption = NULL;
241 uint32 bootstrap_data_checksum_version = 0; /* No checksum */
242 yyscan_t scanner;
243
245
246 InitStandaloneProcess(argv[0]);
247
248 /* Set defaults, to be overridden by explicit options below */
250
251 /* an initial --boot or --check should be present */
252 Assert(argc > 1
253 && (strcmp(argv[1], "--boot") == 0
254 || strcmp(argv[1], "--check") == 0));
255 argv++;
256 argc--;
257
258 while ((flag = getopt(argc, argv, "B:c:d:D:Fkr:X:-:")) != -1)
259 {
260 switch (flag)
261 {
262 case 'B':
263 SetConfigOption("shared_buffers", optarg, PGC_POSTMASTER, PGC_S_ARGV);
264 break;
265 case '-':
266
267 /*
268 * Error if the user misplaced a special must-be-first option
269 * for dispatching to a subprogram. parse_dispatch_option()
270 * returns DISPATCH_POSTMASTER if it doesn't find a match, so
271 * error for anything else.
272 */
276 errmsg("--%s must be first argument", optarg)));
277
279 case 'c':
280 {
281 char *name,
282 *value;
283
285 if (!value)
286 {
287 if (flag == '-')
290 errmsg("--%s requires a value",
291 optarg)));
292 else
295 errmsg("-c %s requires a value",
296 optarg)));
297 }
298
300 pfree(name);
301 pfree(value);
302 break;
303 }
304 case 'D':
306 break;
307 case 'd':
308 {
309 /* Turn on debugging for the bootstrap process. */
310 char *debugstr;
311
312 debugstr = psprintf("debug%s", optarg);
313 SetConfigOption("log_min_messages", debugstr,
315 SetConfigOption("client_min_messages", debugstr,
318 }
319 break;
320 case 'F':
321 SetConfigOption("fsync", "false", PGC_POSTMASTER, PGC_S_ARGV);
322 break;
323 case 'k':
325 break;
326 case 'r':
328 break;
329 case 'X':
331 break;
332 default:
333 write_stderr("Try \"%s --help\" for more information.\n",
334 progname);
335 proc_exit(1);
336 break;
337 }
338 }
339
340 if (argc != optind)
341 {
342 write_stderr("%s: invalid command-line arguments\n", progname);
343 proc_exit(1);
344 }
345
346 /* Acquire configuration parameters */
348 proc_exit(1);
349
350 /*
351 * Validate we have been given a reasonable-looking DataDir and change
352 * into it
353 */
354 checkDataDir();
356
358
360 IgnoreSystemIndexes = true;
361
363
364 /*
365 * Even though bootstrapping runs in single-process mode, initialize
366 * postmaster child slots array so that --check can detect running out of
367 * shared memory or other resources if max_connections is set too high.
368 */
370
372
374
375 /*
376 * Estimate number of openable files. This is essential too in --check
377 * mode, because on some platforms semaphores count as open files.
378 */
380
381 /*
382 * XXX: It might make sense to move this into its own function at some
383 * point. Right now it seems like it'd cause more code duplication than
384 * it's worth.
385 */
386 if (check_only)
387 {
390 abort();
391 }
392
393 /*
394 * Do backend-like initialization for bootstrap mode
395 */
396 InitProcess();
397
398 BaseInit();
399
402
403 /*
404 * To ensure that src/common/link-canary.c is linked into the backend, we
405 * must call it from somewhere. Here is as good as anywhere.
406 */
408 elog(ERROR, "backend is incorrectly linked to frontend functions");
409
411
412 /* Initialize stuff for bootstrap-file processing */
413 for (i = 0; i < MAXATTR; i++)
414 {
415 attrtypes[i] = NULL;
416 Nulls[i] = false;
417 }
418
419 if (boot_yylex_init(&scanner) != 0)
420 elog(ERROR, "yylex_init() failed: %m");
421
422 /*
423 * Process bootstrap input.
424 */
426 boot_yyparse(scanner);
428
429 /*
430 * We should now know about all mapped relations, so it's okay to write
431 * out the initial relation mapping files.
432 */
434
435 /* Clean up and exit */
436 cleanup();
437 proc_exit(0);
438}
439
440
441/* ----------------------------------------------------------------
442 * misc functions
443 * ----------------------------------------------------------------
444 */
445
446/*
447 * Set up signal handling for a bootstrap process
448 */
449static void
451{
453
454 /*
455 * We don't actually need any non-default signal handling in bootstrap
456 * mode; "curl up and die" is a sufficient response for all these cases.
457 * Let's set that handling explicitly, as documentation if nothing else.
458 */
463}
464
465/* ----------------------------------------------------------------
466 * MANUAL BACKEND INTERACTIVE INTERFACE COMMANDS
467 * ----------------------------------------------------------------
468 */
469
470/* ----------------
471 * boot_openrel
472 *
473 * Execute BKI OPEN command.
474 * ----------------
475 */
476void
478{
479 int i;
480
481 if (strlen(relname) >= NAMEDATALEN)
482 relname[NAMEDATALEN - 1] = '\0';
483
484 /*
485 * pg_type must be filled before any OPEN command is executed, hence we
486 * can now populate Typ if we haven't yet.
487 */
488 if (Typ == NIL)
490
491 if (boot_reldesc != NULL)
492 closerel(NULL);
493
494 elog(DEBUG4, "open relation %s, attrsize %d",
496
499 for (i = 0; i < numattr; i++)
500 {
501 if (attrtypes[i] == NULL)
506
507 {
509
510 elog(DEBUG4, "create attribute %d name %s len %d num %d type %u",
511 i, NameStr(at->attname), at->attlen, at->attnum,
512 at->atttypid);
513 }
514 }
515}
516
517/* ----------------
518 * closerel
519 * ----------------
520 */
521void
523{
524 if (relname)
525 {
526 if (boot_reldesc)
527 {
529 elog(ERROR, "close of %s when %s was expected",
531 }
532 else
533 elog(ERROR, "close of %s before any relation was opened",
534 relname);
535 }
536
537 if (boot_reldesc == NULL)
538 elog(ERROR, "no open relation to close");
539 else
540 {
541 elog(DEBUG4, "close relation %s",
545 }
546}
547
548
549
550/* ----------------
551 * DEFINEATTR()
552 *
553 * define a <field,type> pair
554 * if there are n fields in a relation to be created, this routine
555 * will be called n times
556 * ----------------
557 */
558void
559DefineAttr(char *name, char *type, int attnum, int nullness)
560{
561 Oid typeoid;
562
563 if (boot_reldesc != NULL)
564 {
565 elog(WARNING, "no open relations allowed with CREATE command");
566 closerel(NULL);
567 }
568
569 if (attrtypes[attnum] == NULL)
572
574 elog(DEBUG4, "column %s %s", NameStr(attrtypes[attnum]->attname), type);
575 attrtypes[attnum]->attnum = attnum + 1;
576
577 typeoid = gettype(type);
578
579 if (Typ != NIL)
580 {
581 attrtypes[attnum]->atttypid = Ap->am_oid;
582 attrtypes[attnum]->attlen = Ap->am_typ.typlen;
583 attrtypes[attnum]->attbyval = Ap->am_typ.typbyval;
584 attrtypes[attnum]->attalign = Ap->am_typ.typalign;
585 attrtypes[attnum]->attstorage = Ap->am_typ.typstorage;
586 attrtypes[attnum]->attcompression = InvalidCompressionMethod;
587 attrtypes[attnum]->attcollation = Ap->am_typ.typcollation;
588 /* if an array type, assume 1-dimensional attribute */
589 if (Ap->am_typ.typelem != InvalidOid && Ap->am_typ.typlen < 0)
590 attrtypes[attnum]->attndims = 1;
591 else
592 attrtypes[attnum]->attndims = 0;
593 }
594 else
595 {
596 attrtypes[attnum]->atttypid = TypInfo[typeoid].oid;
597 attrtypes[attnum]->attlen = TypInfo[typeoid].len;
598 attrtypes[attnum]->attbyval = TypInfo[typeoid].byval;
599 attrtypes[attnum]->attalign = TypInfo[typeoid].align;
600 attrtypes[attnum]->attstorage = TypInfo[typeoid].storage;
601 attrtypes[attnum]->attcompression = InvalidCompressionMethod;
602 attrtypes[attnum]->attcollation = TypInfo[typeoid].collation;
603 /* if an array type, assume 1-dimensional attribute */
604 if (TypInfo[typeoid].elem != InvalidOid &&
605 attrtypes[attnum]->attlen < 0)
606 attrtypes[attnum]->attndims = 1;
607 else
608 attrtypes[attnum]->attndims = 0;
609 }
610
611 /*
612 * If a system catalog column is collation-aware, force it to use C
613 * collation, so that its behavior is independent of the database's
614 * collation. This is essential to allow template0 to be cloned with a
615 * different database collation.
616 */
617 if (OidIsValid(attrtypes[attnum]->attcollation))
618 attrtypes[attnum]->attcollation = C_COLLATION_OID;
619
620 attrtypes[attnum]->atttypmod = -1;
621 attrtypes[attnum]->attislocal = true;
622
624 {
625 attrtypes[attnum]->attnotnull = true;
626 }
628 {
629 attrtypes[attnum]->attnotnull = false;
630 }
631 else
632 {
634
635 /*
636 * Mark as "not null" if type is fixed-width and prior columns are
637 * likewise fixed-width and not-null. This corresponds to case where
638 * column can be accessed directly via C struct declaration.
639 */
640 if (attrtypes[attnum]->attlen > 0)
641 {
642 int i;
643
644 /* check earlier attributes */
645 for (i = 0; i < attnum; i++)
646 {
647 if (attrtypes[i]->attlen <= 0 ||
649 break;
650 }
651 if (i == attnum)
652 attrtypes[attnum]->attnotnull = true;
653 }
654 }
655}
656
657
658/* ----------------
659 * InsertOneTuple
660 *
661 * If objectid is not zero, it is a specific OID to assign to the tuple.
662 * Otherwise, an OID will be assigned (if necessary) by heap_insert.
663 * ----------------
664 */
665void
667{
668 HeapTuple tuple;
669 TupleDesc tupDesc;
670 int i;
671
672 elog(DEBUG4, "inserting row with %d columns", numattr);
673
675 tuple = heap_form_tuple(tupDesc, values, Nulls);
676 pfree(tupDesc); /* just free's tupDesc, not the attrtypes */
677
679 heap_freetuple(tuple);
680 elog(DEBUG4, "row inserted");
681
682 /*
683 * Reset null markers for next tuple
684 */
685 for (i = 0; i < numattr; i++)
686 Nulls[i] = false;
687}
688
689/* ----------------
690 * InsertOneValue
691 * ----------------
692 */
693void
695{
697 Oid typoid;
698 int16 typlen;
699 bool typbyval;
700 char typalign;
701 char typdelim;
702 Oid typioparam;
704 Oid typoutput;
705 Oid typcollation;
706
707 Assert(i >= 0 && i < MAXATTR);
708
709 elog(DEBUG4, "inserting column %d value \"%s\"", i, value);
710
712 typoid = attr->atttypid;
713
715 &typlen, &typbyval, &typalign,
716 &typdelim, &typioparam,
717 &typinput, &typoutput,
718 &typcollation);
719
720 /*
721 * pg_node_tree values can't be inserted normally (pg_node_tree_in would
722 * just error out), so provide special cases for such columns that we
723 * would like to fill during bootstrap.
724 */
725 if (typoid == PG_NODE_TREEOID)
726 {
727 /* pg_proc.proargdefaults */
731 else /* maybe other cases later */
732 elog(ERROR, "can't handle pg_node_tree input for %s.%s",
734 NameStr(attr->attname));
735 }
736 else
737 {
738 /* Normal case */
739 values[i] = OidInputFunctionCall(typinput, value, typioparam, -1);
740 }
741
742 /*
743 * We use ereport not elog here so that parameters aren't evaluated unless
744 * the message is going to be printed, which generally it isn't
745 */
747 (errmsg_internal("inserted -> %s",
748 OidOutputFunctionCall(typoutput, values[i]))));
749}
750
751/* ----------------
752 * InsertOneProargdefaultsValue
753 *
754 * In general, proargdefaults can be a list of any expressions, but
755 * for bootstrap we only support a list of Const nodes. The input
756 * has the form of a text array, and we feed non-null elements to the
757 * typinput functions for the appropriate parameters.
758 * ----------------
759 */
760static void
762{
763 int pronargs;
767 bool *array_nulls;
768 int array_count;
770 char *nodestring;
771
772 /* The pg_proc columns we need to use must have been filled already */
774 "pronargs must come before proargdefaults");
776 "pronargdefaults must come before proargdefaults");
778 "proargtypes must come before proargdefaults");
780 elog(ERROR, "pronargs must not be null");
782 elog(ERROR, "proargtypes must not be null");
785 Assert(pronargs == proargtypes->dim1);
786
787 /* Parse the input string as an array value, then deconstruct to Datums */
791 Int32GetDatum(-1));
794
795 /* The values should correspond to the last N argtypes */
796 if (array_count > pronargs)
797 elog(ERROR, "too many proargdefaults entries");
798
799 /* Build the List of Const nodes */
801 for (int i = 0; i < array_count; i++)
802 {
803 Oid argtype = proargtypes->values[pronargs - array_count + i];
804 int16 typlen;
805 bool typbyval;
806 char typalign;
807 char typdelim;
808 Oid typioparam;
810 Oid typoutput;
811 Oid typcollation;
812 Datum defval;
813 bool defnull;
815
816 boot_get_type_io_data(argtype,
817 &typlen, &typbyval, &typalign,
818 &typdelim, &typioparam,
819 &typinput, &typoutput,
820 &typcollation);
821
823 if (defnull)
824 defval = (Datum) 0;
825 else
828 typioparam, -1);
829
830 defConst = makeConst(argtype,
831 -1, /* never any typmod */
832 typcollation,
833 typlen,
834 defval,
835 defnull,
836 typbyval);
838 }
839
840 /*
841 * Flatten the List to a node-tree string, then convert to a text datum,
842 * which is the storage representation of pg_node_tree.
843 */
847
848 /*
849 * Hack: fill in pronargdefaults with the right value. This is surely
850 * ugly, but it beats making the programmer do it.
851 */
854}
855
856/* ----------------
857 * InsertOneNull
858 * ----------------
859 */
860void
862{
863 elog(DEBUG4, "inserting column %d NULL", i);
864 Assert(i >= 0 && i < MAXATTR);
865 if (TupleDescAttr(boot_reldesc->rd_att, i)->attnotnull)
866 elog(ERROR,
867 "NULL value specified for not-null column \"%s\" of relation \"%s\"",
871 Nulls[i] = true;
872}
873
874/* ----------------
875 * cleanup
876 * ----------------
877 */
878static void
880{
881 if (boot_reldesc != NULL)
882 closerel(NULL);
883}
884
885/* ----------------
886 * populate_typ_list
887 *
888 * Load the Typ list by reading pg_type.
889 * ----------------
890 */
891static void
893{
894 Relation rel;
895 TableScanDesc scan;
898
899 Assert(Typ == NIL);
900
902 scan = table_beginscan_catalog(rel, 0, NULL);
904 while ((tup = heap_getnext(scan, ForwardScanDirection)) != NULL)
905 {
907 struct typmap *newtyp;
908
909 newtyp = palloc_object(struct typmap);
910 Typ = lappend(Typ, newtyp);
911
912 newtyp->am_oid = typForm->oid;
913 memcpy(&newtyp->am_typ, typForm, sizeof(newtyp->am_typ));
914 }
916 table_endscan(scan);
917 table_close(rel, NoLock);
918}
919
920/* ----------------
921 * gettype
922 *
923 * NB: this is really ugly; it will return an integer index into TypInfo[],
924 * and not an OID at all, until the first reference to a type not known in
925 * TypInfo[]. At that point it will read and cache pg_type in Typ,
926 * and subsequently return a real OID (and set the global pointer Ap to
927 * point at the found row in Typ). So caller must check whether Typ is
928 * still NIL to determine what the return value is!
929 * ----------------
930 */
931static Oid
933{
934 if (Typ != NIL)
935 {
936 ListCell *lc;
937
938 foreach(lc, Typ)
939 {
940 struct typmap *app = lfirst(lc);
941
942 if (strncmp(NameStr(app->am_typ.typname), type, NAMEDATALEN) == 0)
943 {
944 Ap = app;
945 return app->am_oid;
946 }
947 }
948
949 /*
950 * The type wasn't known; reload the pg_type contents and check again
951 * to handle composite types, added since last populating the list.
952 */
953
955 Typ = NIL;
957
958 /*
959 * Calling gettype would result in infinite recursion for types
960 * missing in pg_type, so just repeat the lookup.
961 */
962 foreach(lc, Typ)
963 {
964 struct typmap *app = lfirst(lc);
965
966 if (strncmp(NameStr(app->am_typ.typname), type, NAMEDATALEN) == 0)
967 {
968 Ap = app;
969 return app->am_oid;
970 }
971 }
972 }
973 else
974 {
975 int i;
976
977 for (i = 0; i < n_types; i++)
978 {
979 if (strncmp(type, TypInfo[i].name, NAMEDATALEN) == 0)
980 return i;
981 }
982 /* Not in TypInfo, so we'd better be able to read pg_type now */
983 elog(DEBUG4, "external type: %s", type);
985 return gettype(type);
986 }
987 elog(ERROR, "unrecognized type \"%s\"", type);
988 /* not reached, here to make compiler happy */
989 return 0;
990}
991
992/* ----------------
993 * boot_get_type_io_data
994 *
995 * Obtain type I/O information at bootstrap time. This intentionally has
996 * an API very close to that of lsyscache.c's get_type_io_data, except that
997 * we only support obtaining the typinput and typoutput routines, not
998 * the binary I/O routines, and we also return the type's collation.
999 * This is exported so that array_in and array_out can be made to work
1000 * during early bootstrap.
1001 * ----------------
1002 */
1003void
1005 int16 *typlen,
1006 bool *typbyval,
1007 char *typalign,
1008 char *typdelim,
1009 Oid *typioparam,
1010 Oid *typinput,
1011 Oid *typoutput,
1012 Oid *typcollation)
1013{
1014 if (Typ != NIL)
1015 {
1016 /* We have the boot-time contents of pg_type, so use it */
1017 struct typmap *ap = NULL;
1018 ListCell *lc;
1019
1020 foreach(lc, Typ)
1021 {
1022 ap = lfirst(lc);
1023 if (ap->am_oid == typid)
1024 break;
1025 }
1026
1027 if (!ap || ap->am_oid != typid)
1028 elog(ERROR, "type OID %u not found in Typ list", typid);
1029
1030 *typlen = ap->am_typ.typlen;
1031 *typbyval = ap->am_typ.typbyval;
1032 *typalign = ap->am_typ.typalign;
1033 *typdelim = ap->am_typ.typdelim;
1034
1035 /* XXX this logic must match getTypeIOParam() */
1036 if (OidIsValid(ap->am_typ.typelem))
1037 *typioparam = ap->am_typ.typelem;
1038 else
1039 *typioparam = typid;
1040
1041 *typinput = ap->am_typ.typinput;
1042 *typoutput = ap->am_typ.typoutput;
1043
1044 *typcollation = ap->am_typ.typcollation;
1045 }
1046 else
1047 {
1048 /* We don't have pg_type yet, so use the hard-wired TypInfo array */
1049 int typeindex;
1050
1051 for (typeindex = 0; typeindex < n_types; typeindex++)
1052 {
1053 if (TypInfo[typeindex].oid == typid)
1054 break;
1055 }
1056 if (typeindex >= n_types)
1057 elog(ERROR, "type OID %u not found in TypInfo", typid);
1058
1059 *typlen = TypInfo[typeindex].len;
1060 *typbyval = TypInfo[typeindex].byval;
1062 /* We assume typdelim is ',' for all boot-time types */
1063 *typdelim = ',';
1064
1065 /* XXX this logic must match getTypeIOParam() */
1066 if (OidIsValid(TypInfo[typeindex].elem))
1067 *typioparam = TypInfo[typeindex].elem;
1068 else
1069 *typioparam = typid;
1070
1072 *typoutput = TypInfo[typeindex].outproc;
1073
1074 *typcollation = TypInfo[typeindex].collation;
1075 }
1076}
1077
1078/* ----------------
1079 * boot_get_role_oid
1080 *
1081 * Look up a role name at bootstrap time. This is equivalent to
1082 * get_role_oid(rolname, true): return the role OID or InvalidOid if
1083 * not found. We only need to cope with built-in role names.
1084 * ----------------
1085 */
1086Oid
1088{
1089 for (int i = 0; i < lengthof(RolInfo); i++)
1090 {
1091 if (strcmp(RolInfo[i].rolname, rolname) == 0)
1092 return RolInfo[i].oid;
1093 }
1094 return InvalidOid;
1095}
1096
1097/* ----------------
1098 * AllocateAttribute
1099 *
1100 * Note: bootstrap never sets any per-column ACLs, so we only need
1101 * ATTRIBUTE_FIXED_PART_SIZE space per attribute.
1102 * ----------------
1103 */
1104static Form_pg_attribute
1110
1111/*
1112 * index_register() -- record an index that has been set up for building
1113 * later.
1114 *
1115 * At bootstrap time, we define a bunch of indexes on system catalogs.
1116 * We postpone actually building the indexes until just before we're
1117 * finished with initialization, however. This is because the indexes
1118 * themselves have catalog entries, and those have to be included in the
1119 * indexes on those catalogs. Doing it in two phases is the simplest
1120 * way of making sure the indexes have the right contents at the end.
1121 */
1122void
1124 Oid ind,
1125 const IndexInfo *indexInfo)
1126{
1129
1130 /*
1131 * XXX mao 10/31/92 -- don't gc index reldescs, associated info at
1132 * bootstrap time. we'll declare the indexes now, but want to create them
1133 * later.
1134 */
1135
1136 if (nogc == NULL)
1138 "BootstrapNoGC",
1140
1142
1144 newind->il_heap = heap;
1145 newind->il_ind = ind;
1146 newind->il_info = palloc_object(IndexInfo);
1147
1148 memcpy(newind->il_info, indexInfo, sizeof(IndexInfo));
1149 /* expressions will likely be null, but may as well copy it */
1150 newind->il_info->ii_Expressions =
1151 copyObject(indexInfo->ii_Expressions);
1152 newind->il_info->ii_ExpressionsState = NIL;
1153 /* predicate will likely be null, but may as well copy it */
1154 newind->il_info->ii_Predicate =
1155 copyObject(indexInfo->ii_Predicate);
1156 newind->il_info->ii_PredicateState = NULL;
1157 /* no exclusion constraints at bootstrap time, so no need to copy */
1158 Assert(indexInfo->ii_ExclusionOps == NULL);
1159 Assert(indexInfo->ii_ExclusionProcs == NULL);
1160 Assert(indexInfo->ii_ExclusionStrats == NULL);
1161
1162 newind->il_next = ILHead;
1163 ILHead = newind;
1164
1166}
1167
1168
1169/*
1170 * build_indices -- fill in all the indexes registered earlier
1171 */
1172void
1174{
1175 for (; ILHead != NULL; ILHead = ILHead->il_next)
1176 {
1177 Relation heap;
1178 Relation ind;
1179
1180 /* need not bother with locks during bootstrap */
1181 heap = table_open(ILHead->il_heap, NoLock);
1183
1184 index_build(heap, ind, ILHead->il_info, false, false);
1185
1187 table_close(heap, NoLock);
1188 }
1189}
#define DatumGetArrayTypeP(X)
Definition array.h:261
void deconstruct_array_builtin(const ArrayType *array, Oid elmtype, Datum **elemsp, bool **nullsp, int *nelemsp)
#define write_stderr(str)
Definition parallel.c:186
static Oid gettype(char *type)
Definition bootstrap.c:932
static Datum values[MAXATTR]
Definition bootstrap.c:188
void closerel(char *relname)
Definition bootstrap.c:522
static void populate_typ_list(void)
Definition bootstrap.c:892
static void CheckerModeMain(void)
Definition bootstrap.c:217
void build_indices(void)
Definition bootstrap.c:1173
void InsertOneValue(char *value, int i)
Definition bootstrap.c:694
static void cleanup(void)
Definition bootstrap.c:879
static const int n_types
Definition bootstrap.c:140
void InsertOneNull(int i)
Definition bootstrap.c:861
static const struct rolinfo RolInfo[]
Definition bootstrap.c:167
static const struct typinfo TypInfo[]
Definition bootstrap.c:91
Relation boot_reldesc
Definition bootstrap.c:62
static struct typmap * Ap
Definition bootstrap.c:149
static List * Typ
Definition bootstrap.c:148
void InsertOneTuple(void)
Definition bootstrap.c:666
void boot_get_type_io_data(Oid typid, int16 *typlen, bool *typbyval, char *typalign, char *typdelim, Oid *typioparam, Oid *typinput, Oid *typoutput, Oid *typcollation)
Definition bootstrap.c:1004
static void InsertOneProargdefaultsValue(char *value)
Definition bootstrap.c:761
struct _IndexList IndexList
static void bootstrap_signals(void)
Definition bootstrap.c:450
Form_pg_attribute attrtypes[MAXATTR]
Definition bootstrap.c:64
Oid boot_get_role_oid(const char *rolname)
Definition bootstrap.c:1087
int numattr
Definition bootstrap.c:65
static Form_pg_attribute AllocateAttribute(void)
Definition bootstrap.c:1105
static MemoryContext nogc
Definition bootstrap.c:191
void BootstrapModeMain(int argc, char *argv[], bool check_only)
Definition bootstrap.c:235
void DefineAttr(char *name, char *type, int attnum, int nullness)
Definition bootstrap.c:559
static bool Nulls[MAXATTR]
Definition bootstrap.c:189
void index_register(Oid heap, Oid ind, const IndexInfo *indexInfo)
Definition bootstrap.c:1123
void boot_openrel(char *relname)
Definition bootstrap.c:477
static IndexList * ILHead
Definition bootstrap.c:207
#define BOOTCOL_NULL_FORCE_NULL
Definition bootstrap.h:29
int boot_yylex_init(yyscan_t *yyscannerp)
#define MAXATTR
Definition bootstrap.h:26
#define BOOTCOL_NULL_FORCE_NOT_NULL
Definition bootstrap.h:30
#define BOOTCOL_NULL_AUTO
Definition bootstrap.h:28
int boot_yyparse(yyscan_t yyscanner)
#define PG_DATA_CHECKSUM_VERSION
Definition bufpage.h:232
#define CStringGetTextDatum(s)
Definition builtins.h:98
#define NameStr(name)
Definition c.h:837
#define Assert(condition)
Definition c.h:945
int16_t int16
Definition c.h:613
uint32_t uint32
Definition c.h:618
#define lengthof(array)
Definition c.h:875
#define pg_fallthrough
Definition c.h:152
#define MemSet(start, val, len)
Definition c.h:1109
#define StaticAssertDecl(condition, errmessage)
Definition c.h:1010
#define OidIsValid(objectId)
Definition c.h:860
void * yyscan_t
Definition cubedata.h:65
int errcode(int sqlerrcode)
Definition elog.c:874
int int errmsg_internal(const char *fmt,...) pg_attribute_printf(1
#define WARNING
Definition elog.h:36
#define ERROR
Definition elog.h:39
#define elog(elevel,...)
Definition elog.h:226
#define ereport(elevel,...)
Definition elog.h:150
#define DEBUG4
Definition elog.h:27
void set_max_safe_fds(void)
Definition fd.c:1045
#define palloc_object(type)
Definition fe_memutils.h:74
Datum OidInputFunctionCall(Oid functionId, char *str, Oid typioparam, int32 typmod)
Definition fmgr.c:1755
char * OidOutputFunctionCall(Oid functionId, Datum val)
Definition fmgr.c:1764
#define OidFunctionCall3(functionId, arg1, arg2, arg3)
Definition fmgr.h:726
bool IsUnderPostmaster
Definition globals.c:120
char OutputFileName[MAXPGPATH]
Definition globals.c:79
void SetConfigOption(const char *name, const char *value, GucContext context, GucSource source)
Definition guc.c:4228
bool SelectConfigFiles(const char *userDoption, const char *progname)
Definition guc.c:1656
void ParseLongOption(const char *string, char **name, char **value)
Definition guc.c:6237
void InitializeGUCOptions(void)
Definition guc.c:1408
@ PGC_S_DYNAMIC_DEFAULT
Definition guc.h:114
@ PGC_S_ARGV
Definition guc.h:117
@ PGC_INTERNAL
Definition guc.h:73
@ PGC_POSTMASTER
Definition guc.h:74
HeapTuple heap_getnext(TableScanDesc sscan, ScanDirection direction)
Definition heapam.c:1420
void simple_heap_insert(Relation relation, HeapTuple tup)
Definition heapam.c:2796
HeapTuple heap_form_tuple(TupleDesc tupleDescriptor, const Datum *values, const bool *isnull)
Definition heaptuple.c:1037
void heap_freetuple(HeapTuple htup)
Definition heaptuple.c:1384
static void * GETSTRUCT(const HeapTupleData *tuple)
void index_build(Relation heapRelation, Relation indexRelation, IndexInfo *indexInfo, bool isreindex, bool parallel)
Definition index.c:3003
void index_close(Relation relation, LOCKMODE lockmode)
Definition indexam.c:177
Relation index_open(Oid relationId, LOCKMODE lockmode)
Definition indexam.c:133
static struct @174 value
void proc_exit(int code)
Definition ipc.c:105
void CreateSharedMemoryAndSemaphores(void)
Definition ipci.c:192
int i
Definition isn.c:77
List * lappend(List *list, void *datum)
Definition list.c:339
void list_free_deep(List *list)
Definition list.c:1560
#define NoLock
Definition lockdefs.h:34
DispatchOption parse_dispatch_option(const char *name)
Definition main.c:244
const char * progname
Definition main.c:44
RangeVar * makeRangeVar(char *schemaname, char *relname, int location)
Definition makefuncs.c:473
Const * makeConst(Oid consttype, int32 consttypmod, Oid constcollid, int constlen, Datum constvalue, bool constisnull, bool constbyval)
Definition makefuncs.c:350
void * MemoryContextAllocZero(MemoryContext context, Size size)
Definition mcxt.c:1266
char * pstrdup(const char *in)
Definition mcxt.c:1781
void pfree(void *pointer)
Definition mcxt.c:1616
MemoryContext TopMemoryContext
Definition mcxt.c:166
#define AllocSetContextCreate
Definition memutils.h:129
#define ALLOCSET_DEFAULT_SIZES
Definition memutils.h:160
@ NormalProcessing
Definition miscadmin.h:472
@ BootstrapProcessing
Definition miscadmin.h:470
#define SetProcessingMode(mode)
Definition miscadmin.h:483
void ChangeToDataDir(void)
Definition miscinit.c:410
void InitStandaloneProcess(const char *argv0)
Definition miscinit.c:176
bool IgnoreSystemIndexes
Definition miscinit.c:82
void checkDataDir(void)
Definition miscinit.c:297
void CreateDataDirLockFile(bool amPostmaster)
Definition miscinit.c:1464
void namestrcpy(Name name, const char *str)
Definition name.c:233
#define copyObject(obj)
Definition nodes.h:232
static char * errmsg
char * nodeToString(const void *obj)
Definition outfuncs.c:811
static MemoryContext MemoryContextSwitchTo(MemoryContext context)
Definition palloc.h:124
NameData attname
#define ATTRIBUTE_FIXED_PART_SIZE
int16 attnum
int16 attlen
FormData_pg_attribute * Form_pg_attribute
bool attnotnull
NameData rolname
Definition pg_authid.h:36
NameData relname
Definition pg_class.h:40
#define NAMEDATALEN
#define MAXPGPATH
PGDLLIMPORT int optind
Definition getopt.c:51
int getopt(int nargc, char *const *nargv, const char *ostr)
Definition getopt.c:72
PGDLLIMPORT char * optarg
Definition getopt.c:53
#define lfirst(lc)
Definition pg_list.h:172
#define NIL
Definition pg_list.h:68
int16 pronargs
Definition pg_proc.h:83
END_CATALOG_STRUCT typedef FormData_pg_type * Form_pg_type
Definition pg_type.h:265
char typalign
Definition pg_type.h:178
FormData_pg_type
Definition pg_type.h:256
void InitPostmasterChildSlots(void)
Definition pmchild.c:97
#define pqsignal
Definition port.h:547
size_t strlcpy(char *dst, const char *src, size_t siz)
Definition strlcpy.c:45
static const char * userDoption
Definition postgres.c:156
static Datum PointerGetDatum(const void *X)
Definition postgres.h:342
static Datum Int16GetDatum(int16 X)
Definition postgres.h:172
static Datum ObjectIdGetDatum(Oid X)
Definition postgres.h:252
static char * DatumGetCString(Datum X)
Definition postgres.h:355
uint64_t Datum
Definition postgres.h:70
static Pointer DatumGetPointer(Datum X)
Definition postgres.h:332
static Datum CStringGetDatum(const char *X)
Definition postgres.h:370
static Datum Int32GetDatum(int32 X)
Definition postgres.h:212
static int16 DatumGetInt16(Datum X)
Definition postgres.h:162
#define InvalidOid
unsigned int Oid
void InitializeMaxBackends(void)
Definition postinit.c:559
void BaseInit(void)
Definition postinit.c:616
void InitializeFastPathLocks(void)
Definition postinit.c:584
void InitPostgres(const char *in_dbname, Oid dboid, const char *username, Oid useroid, bits32 flags, char *out_dbname)
Definition postinit.c:719
@ DISPATCH_POSTMASTER
Definition postmaster.h:139
static int fb(int x)
char * psprintf(const char *fmt,...)
Definition psprintf.c:43
#define RelationGetRelid(relation)
Definition rel.h:514
#define RelationGetDescr(relation)
Definition rel.h:540
#define RelationGetNumberOfAttributes(relation)
Definition rel.h:520
#define RelationGetRelationName(relation)
Definition rel.h:548
void RelationMapFinishBootstrap(void)
Definition relmapper.c:626
@ ForwardScanDirection
Definition sdir.h:28
void InitProcess(void)
Definition proc.c:380
uint16 * ii_ExclusionStrats
Definition execnodes.h:203
Oid * ii_ExclusionOps
Definition execnodes.h:199
List * ii_Expressions
Definition execnodes.h:189
Oid * ii_ExclusionProcs
Definition execnodes.h:201
List * ii_Predicate
Definition execnodes.h:194
Definition pg_list.h:54
TupleDesc rd_att
Definition rel.h:112
struct _IndexList * il_next
Definition bootstrap.c:204
IndexInfo * il_info
Definition bootstrap.c:203
Definition c.h:817
const char * rolname
Definition bootstrap.c:163
Oid oid
Definition bootstrap.c:164
char align
Definition bootstrap.c:84
Oid outproc
Definition bootstrap.c:88
int16 len
Definition bootstrap.c:82
Oid oid
Definition bootstrap.c:80
bool byval
Definition bootstrap.c:83
char name[NAMEDATALEN]
Definition bootstrap.c:79
Oid collation
Definition bootstrap.c:86
Oid elem
Definition bootstrap.c:81
char storage
Definition bootstrap.c:85
Oid inproc
Definition bootstrap.c:87
Oid am_oid
Definition bootstrap.c:144
FormData_pg_type am_typ
Definition bootstrap.c:145
void table_close(Relation relation, LOCKMODE lockmode)
Definition table.c:126
Relation table_open(Oid relationId, LOCKMODE lockmode)
Definition table.c:40
Relation table_openrv(const RangeVar *relation, LOCKMODE lockmode)
Definition table.c:83
TableScanDesc table_beginscan_catalog(Relation relation, int nkeys, ScanKeyData *key)
Definition tableam.c:113
static void table_endscan(TableScanDesc scan)
Definition tableam.h:1004
char * flag(int b)
Definition test-ctype.c:33
#define InvalidCompressionMethod
TupleDesc CreateTupleDesc(int natts, Form_pg_attribute *attrs)
Definition tupdesc.c:216
static FormData_pg_attribute * TupleDescAttr(TupleDesc tupdesc, int i)
Definition tupdesc.h:178
const char * type
const char * name
#define SIGHUP
Definition win32_port.h:158
#define SIGQUIT
Definition win32_port.h:159
void StartTransactionCommand(void)
Definition xact.c:3081
void CommitTransactionCommand(void)
Definition xact.c:3179
void BootStrapXLOG(uint32 data_checksum_version)
Definition xlog.c:5110