mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-25 08:58:14 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			3232 lines
		
	
	
	
		
			93 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			3232 lines
		
	
	
	
		
			93 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*****************************************************************************
 | |
| 
 | |
| Copyright (c) 1996, 2016, Oracle and/or its affiliates. All Rights Reserved.
 | |
| Copyright (c) 2016, 2023, MariaDB Corporation.
 | |
| 
 | |
| This program is free software; you can redistribute it and/or modify it under
 | |
| the terms of the GNU General Public License as published by the Free Software
 | |
| Foundation; version 2 of the License.
 | |
| 
 | |
| This program is distributed in the hope that it will be useful, but WITHOUT
 | |
| ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
 | |
| FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
 | |
| 
 | |
| You should have received a copy of the GNU General Public License along with
 | |
| this program; if not, write to the Free Software Foundation, Inc.,
 | |
| 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA
 | |
| 
 | |
| *****************************************************************************/
 | |
| 
 | |
| /**************************************************//**
 | |
| @file dict/dict0load.cc
 | |
| Loads to the memory cache database object definitions
 | |
| from dictionary tables
 | |
| 
 | |
| Created 4/24/1996 Heikki Tuuri
 | |
| *******************************************************/
 | |
| 
 | |
| #include "dict0load.h"
 | |
| 
 | |
| #include "log.h"
 | |
| #include "btr0pcur.h"
 | |
| #include "btr0btr.h"
 | |
| #include "dict0boot.h"
 | |
| #include "dict0crea.h"
 | |
| #include "dict0dict.h"
 | |
| #include "dict0stats.h"
 | |
| #include "ibuf0ibuf.h"
 | |
| #include "fsp0file.h"
 | |
| #include "fts0priv.h"
 | |
| #include "mach0data.h"
 | |
| #include "page0page.h"
 | |
| #include "rem0cmp.h"
 | |
| #include "srv0start.h"
 | |
| #include "srv0srv.h"
 | |
| #include "fts0opt.h"
 | |
| #include "row0vers.h"
 | |
| 
 | |
| /** Loads a table definition and also all its index definitions.
 | |
| 
 | |
| Loads those foreign key constraints whose referenced table is already in
 | |
| dictionary cache.  If a foreign key constraint is not loaded, then the
 | |
| referenced table is pushed into the output stack (fk_tables), if it is not
 | |
| NULL.  These tables must be subsequently loaded so that all the foreign
 | |
| key constraints are loaded into memory.
 | |
| 
 | |
| @param[in]	name		Table name in the db/tablename format
 | |
| @param[in]	ignore_err	Error to be ignored when loading table
 | |
| 				and its index definition
 | |
| @param[out]	fk_tables	Related table names that must also be
 | |
| 				loaded to ensure that all foreign key
 | |
| 				constraints are loaded.
 | |
| @return table, possibly with file_unreadable flag set
 | |
| @retval nullptr if the table does not exist */
 | |
| static dict_table_t *dict_load_table_one(const span<const char> &name,
 | |
|                                          dict_err_ignore_t ignore_err,
 | |
|                                          dict_names_t &fk_tables);
 | |
| 
 | |
| /** Load an index definition from a SYS_INDEXES record to dict_index_t.
 | |
| @return	error message
 | |
| @retval	NULL on success */
 | |
| static
 | |
| const char*
 | |
| dict_load_index_low(
 | |
| 	byte*		table_id,	/*!< in/out: table id (8 bytes),
 | |
| 					an "in" value if mtr
 | |
| 					and "out" when !mtr */
 | |
| 	bool		uncommitted,	/*!< in: false=READ COMMITTED,
 | |
| 					true=READ UNCOMMITTED */
 | |
| 	mem_heap_t*	heap,		/*!< in/out: temporary memory heap */
 | |
| 	const rec_t*	rec,		/*!< in: SYS_INDEXES record */
 | |
| 	mtr_t*		mtr,		/*!< in/out: mini-transaction,
 | |
| 					or nullptr if a pre-allocated
 | |
| 					*index is to be filled in */
 | |
| 	dict_table_t*	table,		/*!< in/out: table, or NULL */
 | |
| 	dict_index_t**	index);		/*!< out,own: index, or NULL */
 | |
| 
 | |
| /** Load a table column definition from a SYS_COLUMNS record to dict_table_t.
 | |
| @param table           table, or nullptr if the output will be in column
 | |
| @param use_uncommitted 0=READ COMMITTED, 1=detect, 2=READ UNCOMMITTED
 | |
| @param heap            memory heap for temporary storage
 | |
| @param column          pointer to output buffer, or nullptr if table!=nullptr
 | |
| @param table_id        table identifier
 | |
| @param col_name        column name
 | |
| @param rec             SYS_COLUMNS record
 | |
| @param mtr             mini-transaction
 | |
| @param nth_v_col       nullptr, or pointer to a counter of virtual columns
 | |
| @return error message
 | |
| @retval nullptr on success */
 | |
| static const char *dict_load_column_low(dict_table_t *table,
 | |
|                                         unsigned use_uncommitted,
 | |
|                                         mem_heap_t *heap, dict_col_t *column,
 | |
|                                         table_id_t *table_id,
 | |
|                                         const char **col_name,
 | |
|                                         const rec_t *rec,
 | |
|                                         mtr_t *mtr,
 | |
|                                         ulint *nth_v_col);
 | |
| 
 | |
| /** Load a virtual column "mapping" (to base columns) information
 | |
| from a SYS_VIRTUAL record
 | |
| @param[in,out]	table		table
 | |
| @param[in]	uncommitted	false=READ COMMITTED, true=READ UNCOMMITTED
 | |
| @param[in,out]	column		mapped base column's dict_column_t
 | |
| @param[in,out]	table_id	table id
 | |
| @param[in,out]	pos		virtual column position
 | |
| @param[in,out]	base_pos	base column position
 | |
| @param[in]	rec		SYS_VIRTUAL record
 | |
| @return	error message
 | |
| @retval	NULL on success */
 | |
| static
 | |
| const char*
 | |
| dict_load_virtual_low(
 | |
| 	dict_table_t*	table,
 | |
| 	bool		uncommitted,
 | |
| 	dict_col_t**	column,
 | |
| 	table_id_t*	table_id,
 | |
| 	ulint*		pos,
 | |
| 	ulint*		base_pos,
 | |
| 	const rec_t*	rec);
 | |
| 
 | |
| /** Load an index field definition from a SYS_FIELDS record to dict_index_t.
 | |
| @return	error message
 | |
| @retval	NULL on success */
 | |
| static
 | |
| const char*
 | |
| dict_load_field_low(
 | |
| 	byte*		index_id,	/*!< in/out: index id (8 bytes)
 | |
| 					an "in" value if index != NULL
 | |
| 					and "out" if index == NULL */
 | |
| 	bool		uncommitted,	/*!< in: false=READ COMMITTED,
 | |
| 					true=READ UNCOMMITTED */
 | |
| 	dict_index_t*	index,		/*!< in/out: index, could be NULL
 | |
| 					if we just populate a dict_field_t
 | |
| 					struct with information from
 | |
| 					a SYS_FIELDS record */
 | |
| 	dict_field_t*	sys_field,	/*!< out: dict_field_t to be
 | |
| 					filled */
 | |
| 	ulint*		pos,		/*!< out: Field position */
 | |
| 	byte*		last_index_id,	/*!< in: last index id */
 | |
| 	mem_heap_t*	heap,		/*!< in/out: memory heap
 | |
| 					for temporary storage */
 | |
| 	mtr_t*		mtr,		/*!< in/out: mini-transaction */
 | |
| 	const rec_t*	rec);		/*!< in: SYS_FIELDS record */
 | |
| 
 | |
| #ifdef UNIV_DEBUG
 | |
| /****************************************************************//**
 | |
| Compare the name of an index column.
 | |
| @return TRUE if the i'th column of index is 'name'. */
 | |
| static
 | |
| ibool
 | |
| name_of_col_is(
 | |
| /*===========*/
 | |
| 	const dict_table_t*	table,	/*!< in: table */
 | |
| 	const dict_index_t*	index,	/*!< in: index */
 | |
| 	ulint			i,	/*!< in: index field offset */
 | |
| 	const char*		name)	/*!< in: name to compare to */
 | |
| {
 | |
| 	ulint	tmp = dict_col_get_no(dict_field_get_col(
 | |
| 					      dict_index_get_nth_field(
 | |
| 						      index, i)));
 | |
| 
 | |
| 	return(strcmp(name, dict_table_get_col_name(table, tmp)) == 0);
 | |
| }
 | |
| #endif /* UNIV_DEBUG */
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function gets the next system table record as it scans the table.
 | |
| @return the next record if found, NULL if end of scan */
 | |
| static
 | |
| const rec_t*
 | |
| dict_getnext_system_low(
 | |
| /*====================*/
 | |
| 	btr_pcur_t*	pcur,		/*!< in/out: persistent cursor to the
 | |
| 					record*/
 | |
| 	mtr_t*		mtr)		/*!< in: the mini-transaction */
 | |
| {
 | |
| 	rec_t*	rec = NULL;
 | |
| 
 | |
| 	while (!rec) {
 | |
| 		btr_pcur_move_to_next_user_rec(pcur, mtr);
 | |
| 
 | |
| 		rec = btr_pcur_get_rec(pcur);
 | |
| 
 | |
| 		if (!btr_pcur_is_on_user_rec(pcur)) {
 | |
| 			/* end of index */
 | |
| 			btr_pcur_close(pcur);
 | |
| 
 | |
| 			return(NULL);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* Get a record, let's save the position */
 | |
| 	btr_pcur_store_position(pcur, mtr);
 | |
| 
 | |
| 	return(rec);
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function opens a system table, and returns the first record.
 | |
| @return first record of the system table */
 | |
| const rec_t*
 | |
| dict_startscan_system(
 | |
| /*==================*/
 | |
| 	btr_pcur_t*	pcur,		/*!< out: persistent cursor to
 | |
| 					the record */
 | |
| 	mtr_t*		mtr,		/*!< in: the mini-transaction */
 | |
| 	dict_table_t*	table)		/*!< in: system table */
 | |
| {
 | |
|   btr_pcur_init(pcur);
 | |
|   if (pcur->open_leaf(true, table->indexes.start, BTR_SEARCH_LEAF, mtr) !=
 | |
|       DB_SUCCESS)
 | |
|     return nullptr;
 | |
|   const rec_t *rec;
 | |
|   do
 | |
|     rec= dict_getnext_system_low(pcur, mtr);
 | |
|   while (rec && rec_get_deleted_flag(rec, 0));
 | |
|   return rec;
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function gets the next system table record as it scans the table.
 | |
| @return the next record if found, NULL if end of scan */
 | |
| const rec_t*
 | |
| dict_getnext_system(
 | |
| /*================*/
 | |
| 	btr_pcur_t*	pcur,		/*!< in/out: persistent cursor
 | |
| 					to the record */
 | |
| 	mtr_t*		mtr)		/*!< in: the mini-transaction */
 | |
| {
 | |
|   const rec_t *rec=nullptr;
 | |
|   if (pcur->restore_position(BTR_SEARCH_LEAF, mtr) != btr_pcur_t::CORRUPTED)
 | |
|     do
 | |
|       rec= dict_getnext_system_low(pcur, mtr);
 | |
|     while (rec && rec_get_deleted_flag(rec, 0));
 | |
|   return rec;
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function parses a SYS_INDEXES record and populate a dict_index_t
 | |
| structure with the information from the record. For detail information
 | |
| about SYS_INDEXES fields, please refer to dict_boot() function.
 | |
| @return error message, or NULL on success */
 | |
| const char*
 | |
| dict_process_sys_indexes_rec(
 | |
| /*=========================*/
 | |
| 	mem_heap_t*	heap,		/*!< in/out: heap memory */
 | |
| 	const rec_t*	rec,		/*!< in: current SYS_INDEXES rec */
 | |
| 	dict_index_t*	index,		/*!< out: index to be filled */
 | |
| 	table_id_t*	table_id)	/*!< out: index table id */
 | |
| {
 | |
|   byte buf[8];
 | |
| 
 | |
|   ut_d(index->is_dummy = true);
 | |
|   ut_d(index->in_instant_init = false);
 | |
| 
 | |
|   /* Parse the record, and get "dict_index_t" struct filled */
 | |
|   const char *err_msg= dict_load_index_low(buf, false, heap, rec,
 | |
|                                            nullptr, nullptr, &index);
 | |
|   *table_id= mach_read_from_8(buf);
 | |
|   return err_msg;
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function parses a SYS_COLUMNS record and populate a dict_column_t
 | |
| structure with the information from the record.
 | |
| @return error message, or NULL on success */
 | |
| const char*
 | |
| dict_process_sys_columns_rec(
 | |
| /*=========================*/
 | |
| 	mem_heap_t*	heap,		/*!< in/out: heap memory */
 | |
| 	const rec_t*	rec,		/*!< in: current SYS_COLUMNS rec */
 | |
| 	dict_col_t*	column,		/*!< out: dict_col_t to be filled */
 | |
| 	table_id_t*	table_id,	/*!< out: table id */
 | |
| 	const char**	col_name,	/*!< out: column name */
 | |
| 	ulint*		nth_v_col)	/*!< out: if virtual col, this is
 | |
| 					record's sequence number */
 | |
| {
 | |
| 	const char*	err_msg;
 | |
| 
 | |
| 	/* Parse the record, and get "dict_col_t" struct filled */
 | |
| 	err_msg = dict_load_column_low(NULL, 0, heap, column,
 | |
| 				       table_id, col_name, rec, nullptr,
 | |
| 				       nth_v_col);
 | |
| 
 | |
| 	return(err_msg);
 | |
| }
 | |
| 
 | |
| /** This function parses a SYS_VIRTUAL record and extracts virtual column
 | |
| information
 | |
| @param[in]	rec		current SYS_COLUMNS rec
 | |
| @param[in,out]	table_id	table id
 | |
| @param[in,out]	pos		virtual column position
 | |
| @param[in,out]	base_pos	base column position
 | |
| @return error message, or NULL on success */
 | |
| const char*
 | |
| dict_process_sys_virtual_rec(
 | |
| 	const rec_t*	rec,
 | |
| 	table_id_t*	table_id,
 | |
| 	ulint*		pos,
 | |
| 	ulint*		base_pos)
 | |
| {
 | |
|   return dict_load_virtual_low(nullptr, false, nullptr, table_id,
 | |
|                                pos, base_pos, rec);
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function parses a SYS_FIELDS record and populates a dict_field_t
 | |
| structure with the information from the record.
 | |
| @return error message, or NULL on success */
 | |
| const char*
 | |
| dict_process_sys_fields_rec(
 | |
| /*========================*/
 | |
| 	mem_heap_t*	heap,		/*!< in/out: heap memory */
 | |
| 	const rec_t*	rec,		/*!< in: current SYS_FIELDS rec */
 | |
| 	dict_field_t*	sys_field,	/*!< out: dict_field_t to be
 | |
| 					filled */
 | |
| 	ulint*		pos,		/*!< out: Field position */
 | |
| 	index_id_t*	index_id,	/*!< out: current index id */
 | |
| 	index_id_t	last_id)	/*!< in: previous index id */
 | |
| {
 | |
| 	byte		buf[8];
 | |
| 	byte		last_index_id[8];
 | |
| 	const char*	err_msg;
 | |
| 
 | |
| 	mach_write_to_8(last_index_id, last_id);
 | |
| 
 | |
| 	err_msg = dict_load_field_low(buf, false, nullptr, sys_field,
 | |
| 				      pos, last_index_id, heap, nullptr, rec);
 | |
| 
 | |
| 	*index_id = mach_read_from_8(buf);
 | |
| 
 | |
| 	return(err_msg);
 | |
| 
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function parses a SYS_FOREIGN record and populate a dict_foreign_t
 | |
| structure with the information from the record. For detail information
 | |
| about SYS_FOREIGN fields, please refer to dict_load_foreign() function.
 | |
| @return error message, or NULL on success */
 | |
| const char*
 | |
| dict_process_sys_foreign_rec(
 | |
| /*=========================*/
 | |
| 	mem_heap_t*	heap,		/*!< in/out: heap memory */
 | |
| 	const rec_t*	rec,		/*!< in: current SYS_FOREIGN rec */
 | |
| 	dict_foreign_t*	foreign)	/*!< out: dict_foreign_t struct
 | |
| 					to be filled */
 | |
| {
 | |
| 	ulint		len;
 | |
| 	const byte*	field;
 | |
| 
 | |
| 	if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		return("delete-marked record in SYS_FOREIGN");
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_n_fields_old(rec) != DICT_NUM_FIELDS__SYS_FOREIGN) {
 | |
| 		return("wrong number of columns in SYS_FOREIGN record");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__ID, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| err_len:
 | |
| 		return("incorrect column length in SYS_FOREIGN");
 | |
| 	}
 | |
| 
 | |
| 	/* This receives a dict_foreign_t* that points to a stack variable.
 | |
| 	So dict_foreign_free(foreign) is not used as elsewhere.
 | |
| 	Since the heap used here is freed elsewhere, foreign->heap
 | |
| 	is not assigned. */
 | |
| 	foreign->id = mem_heap_strdupl(heap, (const char*) field, len);
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__DB_TRX_ID, &len);
 | |
| 	if (len != DATA_TRX_ID_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__DB_ROLL_PTR, &len);
 | |
| 	if (len != DATA_ROLL_PTR_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	/* The _lookup versions of the referenced and foreign table names
 | |
| 	 are not assigned since they are not used in this dict_foreign_t */
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__FOR_NAME, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	foreign->foreign_table_name = mem_heap_strdupl(
 | |
| 		heap, (const char*) field, len);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__REF_NAME, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	foreign->referenced_table_name = mem_heap_strdupl(
 | |
| 		heap, (const char*) field, len);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__N_COLS, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	uint32_t n_fields_and_type = mach_read_from_4(field);
 | |
| 
 | |
| 	foreign->type = n_fields_and_type >> 24 & ((1U << 6) - 1);
 | |
| 	foreign->n_fields = n_fields_and_type & dict_index_t::MAX_N_FIELDS;
 | |
| 
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function parses a SYS_FOREIGN_COLS record and extract necessary
 | |
| information from the record and return to caller.
 | |
| @return error message, or NULL on success */
 | |
| const char*
 | |
| dict_process_sys_foreign_col_rec(
 | |
| /*=============================*/
 | |
| 	mem_heap_t*	heap,		/*!< in/out: heap memory */
 | |
| 	const rec_t*	rec,		/*!< in: current SYS_FOREIGN_COLS rec */
 | |
| 	const char**	name,		/*!< out: foreign key constraint name */
 | |
| 	const char**	for_col_name,	/*!< out: referencing column name */
 | |
| 	const char**	ref_col_name,	/*!< out: referenced column name
 | |
| 					in referenced table */
 | |
| 	ulint*		pos)		/*!< out: column position */
 | |
| {
 | |
| 	ulint		len;
 | |
| 	const byte*	field;
 | |
| 
 | |
| 	if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		return("delete-marked record in SYS_FOREIGN_COLS");
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_n_fields_old(rec) != DICT_NUM_FIELDS__SYS_FOREIGN_COLS) {
 | |
| 		return("wrong number of columns in SYS_FOREIGN_COLS record");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_COLS__ID, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| err_len:
 | |
| 		return("incorrect column length in SYS_FOREIGN_COLS");
 | |
| 	}
 | |
| 	*name = mem_heap_strdupl(heap, (char*) field, len);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_COLS__POS, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	*pos = mach_read_from_4(field);
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_COLS__DB_TRX_ID, &len);
 | |
| 	if (len != DATA_TRX_ID_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_COLS__DB_ROLL_PTR, &len);
 | |
| 	if (len != DATA_ROLL_PTR_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_COLS__FOR_COL_NAME, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	*for_col_name = mem_heap_strdupl(heap, (char*) field, len);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_COLS__REF_COL_NAME, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	*ref_col_name = mem_heap_strdupl(heap, (char*) field, len);
 | |
| 
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /** Check the validity of a SYS_TABLES record
 | |
| Make sure the fields are the right length and that they
 | |
| do not contain invalid contents.
 | |
| @param[in]	rec	SYS_TABLES record
 | |
| @return error message, or NULL on success */
 | |
| static
 | |
| const char*
 | |
| dict_sys_tables_rec_check(
 | |
| 	const rec_t*	rec)
 | |
| {
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	if (rec_get_n_fields_old(rec) != DICT_NUM_FIELDS__SYS_TABLES) {
 | |
| 		return("wrong number of columns in SYS_TABLES record");
 | |
| 	}
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__NAME, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| err_len:
 | |
| 		return("incorrect column length in SYS_TABLES");
 | |
| 	}
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__DB_TRX_ID, &len);
 | |
| 	if (len != DATA_TRX_ID_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__DB_ROLL_PTR, &len);
 | |
| 	if (len != DATA_ROLL_PTR_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(rec, DICT_FLD__SYS_TABLES__ID, &len);
 | |
| 	if (len != 8) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__N_COLS, &len);
 | |
| 	if (field == NULL || len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(rec, DICT_FLD__SYS_TABLES__TYPE, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__MIX_ID, &len);
 | |
| 	if (len != 8) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__MIX_LEN, &len);
 | |
| 	if (field == NULL || len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__CLUSTER_ID, &len);
 | |
| 	if (len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__SPACE, &len);
 | |
| 	if (field == NULL || len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /** Check if SYS_TABLES.TYPE is valid
 | |
| @param[in]	type		SYS_TABLES.TYPE
 | |
| @param[in]	not_redundant	whether ROW_FORMAT=REDUNDANT is not used
 | |
| @return	whether the SYS_TABLES.TYPE value is valid */
 | |
| static
 | |
| bool
 | |
| dict_sys_tables_type_valid(ulint type, bool not_redundant)
 | |
| {
 | |
| 	/* The DATA_DIRECTORY flag can be assigned fully independently
 | |
| 	of all other persistent table flags. */
 | |
| 	type &= ~DICT_TF_MASK_DATA_DIR;
 | |
| 
 | |
| 	if (type == 1) {
 | |
| 		return(true); /* ROW_FORMAT=REDUNDANT or ROW_FORMAT=COMPACT */
 | |
| 	}
 | |
| 
 | |
| 	if (!(type & 1)) {
 | |
| 		/* For ROW_FORMAT=REDUNDANT and ROW_FORMAT=COMPACT,
 | |
| 		SYS_TABLES.TYPE=1. Else, it is the same as
 | |
| 		dict_table_t::flags, and the least significant bit
 | |
| 		would be set. So, the bit never can be 0. */
 | |
| 		return(false);
 | |
| 	}
 | |
| 
 | |
| 	if (!not_redundant) {
 | |
| 		/* SYS_TABLES.TYPE must be 1 or 1|DICT_TF_MASK_NO_ROLLBACK
 | |
| 		for ROW_FORMAT=REDUNDANT. */
 | |
| 		return !(type & ~(1U | DICT_TF_MASK_NO_ROLLBACK));
 | |
| 	}
 | |
| 
 | |
| 	if (type >= 1U << DICT_TF_POS_UNUSED) {
 | |
| 		/* Some unknown bits are set. */
 | |
| 		return(false);
 | |
| 	}
 | |
| 
 | |
| 	return(dict_tf_is_valid_not_redundant(type));
 | |
| }
 | |
| 
 | |
| /** Convert SYS_TABLES.TYPE to dict_table_t::flags.
 | |
| @param[in]	type		SYS_TABLES.TYPE
 | |
| @param[in]	not_redundant	whether ROW_FORMAT=REDUNDANT is not used
 | |
| @return	table flags */
 | |
| static
 | |
| uint32_t dict_sys_tables_type_to_tf(uint32_t type, bool not_redundant)
 | |
| {
 | |
| 	ut_ad(dict_sys_tables_type_valid(type, not_redundant));
 | |
| 	uint32_t flags = not_redundant ? 1 : 0;
 | |
| 
 | |
| 	/* ZIP_SSIZE, ATOMIC_BLOBS, DATA_DIR, PAGE_COMPRESSION,
 | |
| 	PAGE_COMPRESSION_LEVEL are the same. */
 | |
| 	flags |= type & (DICT_TF_MASK_ZIP_SSIZE
 | |
| 			 | DICT_TF_MASK_ATOMIC_BLOBS
 | |
| 			 | DICT_TF_MASK_DATA_DIR
 | |
| 			 | DICT_TF_MASK_PAGE_COMPRESSION
 | |
| 			 | DICT_TF_MASK_PAGE_COMPRESSION_LEVEL
 | |
| 			 | DICT_TF_MASK_NO_ROLLBACK);
 | |
| 
 | |
| 	ut_ad(dict_tf_is_valid(flags));
 | |
| 	return(flags);
 | |
| }
 | |
| 
 | |
| /** Outcome of dict_sys_tables_rec_read() */
 | |
| enum table_read_status { READ_OK= 0, READ_ERROR, READ_NOT_FOUND };
 | |
| 
 | |
| /** Read and return 5 integer fields from a SYS_TABLES record.
 | |
| @param[in]	rec		A record of SYS_TABLES
 | |
| @param[in]	uncommitted	true=use READ UNCOMMITTED, false=READ COMMITTED
 | |
| @param[in]	mtr		mini-transaction
 | |
| @param[out]	table_id	Pointer to the table_id for this table
 | |
| @param[out]	space_id	Pointer to the space_id for this table
 | |
| @param[out]	n_cols		Pointer to number of columns for this table.
 | |
| @param[out]	flags		Pointer to table flags
 | |
| @param[out]	flags2		Pointer to table flags2
 | |
| @param[out]	trx_id		DB_TRX_ID of the committed SYS_TABLES record,
 | |
| 				or nullptr to perform READ UNCOMMITTED
 | |
| @return whether the record was read correctly */
 | |
| MY_ATTRIBUTE((warn_unused_result))
 | |
| static
 | |
| table_read_status
 | |
| dict_sys_tables_rec_read(
 | |
| 	const rec_t*		rec,
 | |
| 	bool			uncommitted,
 | |
| 	mtr_t*			mtr,
 | |
| 	table_id_t*		table_id,
 | |
| 	uint32_t*		space_id,
 | |
| 	uint32_t*		n_cols,
 | |
| 	uint32_t*		flags,
 | |
| 	uint32_t*		flags2,
 | |
| 	trx_id_t*		trx_id)
 | |
| {
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 	mem_heap_t*	heap = nullptr;
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__DB_TRX_ID, &len);
 | |
| 	ut_ad(len == 6 || len == UNIV_SQL_NULL);
 | |
| 	trx_id_t id = len == 6 ? trx_read_trx_id(field) : 0;
 | |
| 	if (id && !uncommitted && trx_sys.find(nullptr, id, false)) {
 | |
| 		const auto savepoint = mtr->get_savepoint();
 | |
| 		heap = mem_heap_create(1024);
 | |
| 		dict_index_t* index = UT_LIST_GET_FIRST(
 | |
| 			dict_sys.sys_tables->indexes);
 | |
| 		rec_offs* offsets = rec_get_offsets(
 | |
| 			rec, index, nullptr, true, ULINT_UNDEFINED, &heap);
 | |
| 		const rec_t* old_vers;
 | |
| 		row_vers_build_for_semi_consistent_read(
 | |
| 			nullptr, rec, mtr, index, &offsets, &heap,
 | |
| 			heap, &old_vers, nullptr);
 | |
| 		mtr->rollback_to_savepoint(savepoint);
 | |
| 		rec = old_vers;
 | |
| 		if (!rec) {
 | |
| 			mem_heap_free(heap);
 | |
| 			return READ_NOT_FOUND;
 | |
| 		}
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_TABLES__DB_TRX_ID, &len);
 | |
| 		if (UNIV_UNLIKELY(len != 6)) {
 | |
| 			mem_heap_free(heap);
 | |
| 			return READ_ERROR;
 | |
| 		}
 | |
| 		id = trx_read_trx_id(field);
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		ut_ad(id);
 | |
| 		if (trx_id) {
 | |
| 			return READ_NOT_FOUND;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (trx_id) {
 | |
| 		*trx_id = id;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__ID, &len);
 | |
| 	ut_ad(len == 8);
 | |
| 	*table_id = static_cast<table_id_t>(mach_read_from_8(field));
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__SPACE, &len);
 | |
| 	ut_ad(len == 4);
 | |
| 	*space_id = mach_read_from_4(field);
 | |
| 
 | |
| 	/* Read the 4 byte flags from the TYPE field */
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__TYPE, &len);
 | |
| 	ut_a(len == 4);
 | |
| 	uint32_t type = mach_read_from_4(field);
 | |
| 
 | |
| 	/* Handle MDEV-12873 InnoDB SYS_TABLES.TYPE incompatibility
 | |
| 	for PAGE_COMPRESSED=YES in MariaDB 10.2.2 to 10.2.6.
 | |
| 
 | |
| 	MariaDB 10.2.2 introduced the SHARED_SPACE flag from MySQL 5.7,
 | |
| 	shifting the flags PAGE_COMPRESSION, PAGE_COMPRESSION_LEVEL,
 | |
| 	ATOMIC_WRITES (repurposed to NO_ROLLBACK in 10.3.1) by one bit.
 | |
| 	The SHARED_SPACE flag would always
 | |
| 	be written as 0 by MariaDB, because MariaDB does not support
 | |
| 	CREATE TABLESPACE or CREATE TABLE...TABLESPACE for InnoDB.
 | |
| 
 | |
| 	So, instead of the bits AALLLLCxxxxxxx we would have
 | |
| 	AALLLLC0xxxxxxx if the table was created with MariaDB 10.2.2
 | |
| 	to 10.2.6. (AA=ATOMIC_WRITES, LLLL=PAGE_COMPRESSION_LEVEL,
 | |
| 	C=PAGE_COMPRESSED, xxxxxxx=7 bits that were not moved.)
 | |
| 
 | |
| 	The case LLLLC=00000 is not a problem. The problem is the case
 | |
| 	AALLLL10DB00001 where D is the (mostly ignored) DATA_DIRECTORY
 | |
| 	flag and B is the ATOMIC_BLOBS flag (1 for ROW_FORMAT=DYNAMIC
 | |
| 	and 0 for ROW_FORMAT=COMPACT in this case). Other low-order
 | |
| 	bits must be so, because PAGE_COMPRESSED=YES is only allowed
 | |
| 	for ROW_FORMAT=DYNAMIC and ROW_FORMAT=COMPACT, not for
 | |
| 	ROW_FORMAT=REDUNDANT or ROW_FORMAT=COMPRESSED.
 | |
| 
 | |
| 	Starting with MariaDB 10.2.4, the flags would be
 | |
| 	00LLLL10DB00001, because ATOMIC_WRITES is always written as 0.
 | |
| 
 | |
| 	We will concentrate on the PAGE_COMPRESSION_LEVEL and
 | |
| 	PAGE_COMPRESSED=YES. PAGE_COMPRESSED=NO implies
 | |
| 	PAGE_COMPRESSION_LEVEL=0, and in that case all the affected
 | |
| 	bits will be 0. For PAGE_COMPRESSED=YES, the values 1..9 are
 | |
| 	allowed for PAGE_COMPRESSION_LEVEL. That is, we must interpret
 | |
| 	the bits AALLLL10DB00001 as AALLLL1DB00001.
 | |
| 
 | |
| 	If someone created a table in MariaDB 10.2.2 or 10.2.3 with
 | |
| 	the attribute ATOMIC_WRITES=OFF (value 2) and without
 | |
| 	PAGE_COMPRESSED=YES or PAGE_COMPRESSION_LEVEL, that should be
 | |
| 	rejected. The value ATOMIC_WRITES=ON (1) would look like
 | |
| 	ATOMIC_WRITES=OFF, but it would be ignored starting with
 | |
| 	MariaDB 10.2.4. */
 | |
| 	compile_time_assert(DICT_TF_POS_PAGE_COMPRESSION == 7);
 | |
| 	compile_time_assert(DICT_TF_POS_UNUSED == 14);
 | |
| 
 | |
| 	if ((type & 0x19f) != 0x101) {
 | |
| 		/* The table cannot have been created with MariaDB
 | |
| 		10.2.2 to 10.2.6, because they would write the
 | |
| 		low-order bits of SYS_TABLES.TYPE as 0b10xx00001 for
 | |
| 		PAGE_COMPRESSED=YES. No adjustment is applicable. */
 | |
| 	} else if (type >= 3 << 13) {
 | |
| 		/* 10.2.2 and 10.2.3 write ATOMIC_WRITES less than 3,
 | |
| 		and no other flags above that can be set for the
 | |
| 		SYS_TABLES.TYPE to be in the 10.2.2..10.2.6 format.
 | |
| 		This would in any case be invalid format for 10.2 and
 | |
| 		earlier releases. */
 | |
| 		ut_ad(!dict_sys_tables_type_valid(type, true));
 | |
| 	} else {
 | |
| 		/* SYS_TABLES.TYPE is of the form AALLLL10DB00001.  We
 | |
| 		must still validate that the LLLL bits are between 0
 | |
| 		and 9 before we can discard the extraneous 0 bit. */
 | |
| 		ut_ad(!DICT_TF_GET_PAGE_COMPRESSION(type));
 | |
| 
 | |
| 		if ((((type >> 9) & 0xf) - 1) < 9) {
 | |
| 			ut_ad(DICT_TF_GET_PAGE_COMPRESSION_LEVEL(type) & 1);
 | |
| 
 | |
| 			type = (type & 0x7fU) | (type >> 1 & ~0x7fU);
 | |
| 
 | |
| 			ut_ad(DICT_TF_GET_PAGE_COMPRESSION(type));
 | |
| 			ut_ad(DICT_TF_GET_PAGE_COMPRESSION_LEVEL(type) >= 1);
 | |
| 			ut_ad(DICT_TF_GET_PAGE_COMPRESSION_LEVEL(type) <= 9);
 | |
| 		} else {
 | |
| 			ut_ad(!dict_sys_tables_type_valid(type, true));
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* The low order bit of SYS_TABLES.TYPE is always set to 1. But in
 | |
| 	dict_table_t::flags the low order bit is used to determine if the
 | |
| 	ROW_FORMAT=REDUNDANT (0) or anything else (1).
 | |
| 	Read the 4 byte N_COLS field and look at the high order bit.  It
 | |
| 	should be set for COMPACT and later.  It should not be set for
 | |
| 	REDUNDANT. */
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_TABLES__N_COLS, &len);
 | |
| 	ut_a(len == 4);
 | |
| 	*n_cols = mach_read_from_4(field);
 | |
| 
 | |
| 	const bool not_redundant = 0 != (*n_cols & DICT_N_COLS_COMPACT);
 | |
| 
 | |
| 	if (!dict_sys_tables_type_valid(type, not_redundant)) {
 | |
| 		sql_print_error("InnoDB: Table %.*s in InnoDB"
 | |
| 				" data dictionary contains invalid flags."
 | |
| 				" SYS_TABLES.TYPE=" UINT32PF
 | |
| 				" SYS_TABLES.N_COLS=" UINT32PF,
 | |
| 				int(rec_get_field_start_offs(rec, 1)), rec,
 | |
| 				type, *n_cols);
 | |
| err_exit:
 | |
| 		if (UNIV_LIKELY_NULL(heap)) {
 | |
| 			mem_heap_free(heap);
 | |
| 		}
 | |
| 		return READ_ERROR;
 | |
| 	}
 | |
| 
 | |
| 	*flags = dict_sys_tables_type_to_tf(type, not_redundant);
 | |
| 
 | |
| 	/* For tables created before MySQL 4.1, there may be
 | |
| 	garbage in SYS_TABLES.MIX_LEN where flags2 are found. Such tables
 | |
| 	would always be in ROW_FORMAT=REDUNDANT which do not have the
 | |
| 	high bit set in n_cols, and flags would be zero.
 | |
| 	MySQL 4.1 was the first version to support innodb_file_per_table,
 | |
| 	that is, *space_id != 0. */
 | |
| 	if (not_redundant || *space_id != 0 || *n_cols & DICT_N_COLS_COMPACT
 | |
| 	    || fil_system.sys_space->full_crc32()) {
 | |
| 
 | |
| 		/* Get flags2 from SYS_TABLES.MIX_LEN */
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_TABLES__MIX_LEN, &len);
 | |
| 		*flags2 = mach_read_from_4(field);
 | |
| 
 | |
| 		if (!dict_tf2_is_valid(*flags, *flags2)) {
 | |
| 			sql_print_error("InnoDB: Table %.*s in InnoDB"
 | |
| 					" data dictionary"
 | |
| 					" contains invalid flags."
 | |
| 					" SYS_TABLES.TYPE=" UINT32PF
 | |
| 					" SYS_TABLES.MIX_LEN=" UINT32PF,
 | |
| 					int(rec_get_field_start_offs(rec, 1)),
 | |
| 					rec,
 | |
| 					type, *flags2);
 | |
| 			goto err_exit;
 | |
| 		}
 | |
| 
 | |
| 		/* DICT_TF2_FTS will be set when indexes are being loaded */
 | |
| 		*flags2 &= ~DICT_TF2_FTS;
 | |
| 
 | |
| 		/* Now that we have used this bit, unset it. */
 | |
| 		*n_cols &= ~DICT_N_COLS_COMPACT;
 | |
| 	} else {
 | |
| 		*flags2 = 0;
 | |
| 	}
 | |
| 
 | |
| 	if (UNIV_LIKELY_NULL(heap)) {
 | |
| 		mem_heap_free(heap);
 | |
| 	}
 | |
| 
 | |
| 	return READ_OK;
 | |
| }
 | |
| 
 | |
| /** @return SELECT MAX(space) FROM sys_tables */
 | |
| static uint32_t dict_find_max_space_id(btr_pcur_t *pcur, mtr_t *mtr)
 | |
| {
 | |
|   uint32_t max_space_id= 0;
 | |
| 
 | |
|   for (const rec_t *rec= dict_startscan_system(pcur, mtr, dict_sys.sys_tables);
 | |
|        rec; rec= dict_getnext_system_low(pcur, mtr))
 | |
|     if (!dict_sys_tables_rec_check(rec))
 | |
|     {
 | |
|       ulint len;
 | |
|       const byte *field=
 | |
|         rec_get_nth_field_old(rec, DICT_FLD__SYS_TABLES__SPACE, &len);
 | |
|       ut_ad(len == 4);
 | |
|       max_space_id= std::max(max_space_id, mach_read_from_4(field));
 | |
|     }
 | |
| 
 | |
|   return max_space_id;
 | |
| }
 | |
| 
 | |
| /** Check MAX(SPACE) FROM SYS_TABLES and store it in fil_system.
 | |
| Open each data file if an encryption plugin has been loaded.
 | |
| 
 | |
| @param spaces  set of tablespace files to open */
 | |
| void dict_check_tablespaces_and_store_max_id(const std::set<uint32_t> *spaces)
 | |
| {
 | |
| 	uint32_t	max_space_id = 0;
 | |
| 	btr_pcur_t	pcur;
 | |
| 	mtr_t		mtr;
 | |
| 
 | |
| 	DBUG_ENTER("dict_check_tablespaces_and_store_max_id");
 | |
| 
 | |
| 	mtr.start();
 | |
| 
 | |
| 	dict_sys.lock(SRW_LOCK_CALL);
 | |
| 
 | |
| 	if (!spaces && ibuf.empty
 | |
| 	    && !encryption_key_id_exists(FIL_DEFAULT_ENCRYPTION_KEY)) {
 | |
| 		max_space_id = dict_find_max_space_id(&pcur, &mtr);
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	for (const rec_t *rec = dict_startscan_system(&pcur, &mtr,
 | |
| 						      dict_sys.sys_tables);
 | |
| 	     rec; rec = dict_getnext_system_low(&pcur, &mtr)) {
 | |
| 		ulint		len;
 | |
| 		table_id_t	table_id;
 | |
| 		uint32_t	space_id;
 | |
| 		uint32_t	n_cols;
 | |
| 		uint32_t	flags;
 | |
| 		uint32_t	flags2;
 | |
| 
 | |
| 		/* If a table record is not useable, ignore it and continue
 | |
| 		on to the next record. Error messages were logged. */
 | |
| 		if (dict_sys_tables_rec_check(rec)) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		const char *field = reinterpret_cast<const char*>(
 | |
| 			rec_get_nth_field_old(rec, DICT_FLD__SYS_TABLES__NAME,
 | |
| 					      &len));
 | |
| 
 | |
| 		DBUG_PRINT("dict_check_sys_tables",
 | |
| 			   ("name: %*.s", static_cast<int>(len), field));
 | |
| 
 | |
| 		if (dict_sys_tables_rec_read(rec, false,
 | |
| 					     &mtr, &table_id, &space_id,
 | |
| 					     &n_cols, &flags, &flags2, nullptr)
 | |
| 		    != READ_OK
 | |
| 		    || space_id == TRX_SYS_SPACE) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		/* For tables or partitions using .ibd files, the flag
 | |
| 		DICT_TF2_USE_FILE_PER_TABLE was not set in MIX_LEN
 | |
| 		before MySQL 5.6.5. The flag should not have been
 | |
| 		introduced in persistent storage. MariaDB will keep
 | |
| 		setting the flag when writing SYS_TABLES entries for
 | |
| 		newly created or rebuilt tables or partitions, but
 | |
| 		will otherwise ignore the flag. */
 | |
| 
 | |
| 		if (fil_space_for_table_exists_in_mem(space_id, flags)) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		if (spaces && spaces->find(uint32_t(space_id))
 | |
|                     == spaces->end()) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		if (flags2 & DICT_TF2_DISCARDED) {
 | |
| 			sql_print_information("InnoDB: Ignoring tablespace"
 | |
| 					      " for %.*s because "
 | |
| 					      "the DISCARD flag is set",
 | |
| 					      static_cast<int>(len), field);
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		const span<const char> name{field, len};
 | |
| 
 | |
| 		char*	filepath = fil_make_filepath(nullptr, name,
 | |
| 						     IBD, false);
 | |
| 
 | |
| 		const bool not_dropped{!rec_get_deleted_flag(rec, 0)};
 | |
| 
 | |
| 		/* Check that the .ibd file exists. */
 | |
| 		if (fil_ibd_open(space_id, dict_tf_to_fsp_flags(flags),
 | |
| 				 not_dropped
 | |
| 				 ? fil_space_t::VALIDATE_NOTHING
 | |
| 				 : fil_space_t::MAYBE_MISSING,
 | |
| 				 name, filepath)) {
 | |
| 		} else if (!not_dropped) {
 | |
| 		} else if (srv_operation == SRV_OPERATION_NORMAL
 | |
| 			   && srv_start_after_restore
 | |
| 			   && srv_force_recovery < SRV_FORCE_NO_BACKGROUND
 | |
| 			   && dict_table_t::is_temporary_name(filepath)) {
 | |
| 			/* Mariabackup will not copy files whose
 | |
| 			names start with #sql-. This table ought to
 | |
| 			be dropped by drop_garbage_tables_after_restore()
 | |
| 			a little later. */
 | |
| 		} else {
 | |
| 			sql_print_warning("InnoDB: Ignoring tablespace for"
 | |
| 					  " %.*s because it"
 | |
| 					  " could not be opened.",
 | |
| 					  static_cast<int>(len), field);
 | |
| 		}
 | |
| 
 | |
| 		max_space_id = ut_max(max_space_id, space_id);
 | |
| 
 | |
| 		ut_free(filepath);
 | |
| 	}
 | |
| 
 | |
| done:
 | |
| 	mtr.commit();
 | |
| 
 | |
| 	fil_set_max_space_id_if_bigger(max_space_id);
 | |
| 
 | |
| 	dict_sys.unlock();
 | |
| 
 | |
| 	DBUG_VOID_RETURN;
 | |
| }
 | |
| 
 | |
| /** Error message for a delete-marked record in dict_load_column_low() */
 | |
| static const char *dict_load_column_del= "delete-marked record in SYS_COLUMNS";
 | |
| /** Error message for a missing record in dict_load_column_low() */
 | |
| static const char *dict_load_column_none= "SYS_COLUMNS record not found";
 | |
| /** Message for incomplete instant ADD/DROP in dict_load_column_low() */
 | |
| static const char *dict_load_column_instant= "incomplete instant ADD/DROP";
 | |
| 
 | |
| /** Load a table column definition from a SYS_COLUMNS record to dict_table_t.
 | |
| @param table           table, or nullptr if the output will be in column
 | |
| @param use_uncommitted 0=READ COMMITTED, 1=detect, 2=READ UNCOMMITTED
 | |
| @param heap            memory heap for temporary storage
 | |
| @param column          pointer to output buffer, or nullptr if table!=nullptr
 | |
| @param table_id        table identifier
 | |
| @param col_name        column name
 | |
| @param rec             SYS_COLUMNS record
 | |
| @param mtr             mini-transaction
 | |
| @param nth_v_col       nullptr, or pointer to a counter of virtual columns
 | |
| @return error message
 | |
| @retval nullptr on success */
 | |
| static const char *dict_load_column_low(dict_table_t *table,
 | |
|                                         unsigned use_uncommitted,
 | |
|                                         mem_heap_t *heap, dict_col_t *column,
 | |
|                                         table_id_t *table_id,
 | |
|                                         const char **col_name,
 | |
|                                         const rec_t *rec,
 | |
|                                         mtr_t *mtr,
 | |
|                                         ulint *nth_v_col)
 | |
| {
 | |
| 	char*		name;
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 	ulint		mtype;
 | |
| 	ulint		prtype;
 | |
| 	ulint		col_len;
 | |
| 	ulint		pos;
 | |
| 	ulint		num_base;
 | |
| 
 | |
| 	ut_ad(!table == !!column);
 | |
| 
 | |
| 	if (rec_get_n_fields_old(rec) != DICT_NUM_FIELDS__SYS_COLUMNS) {
 | |
| 		return("wrong number of columns in SYS_COLUMNS record");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__TABLE_ID, &len);
 | |
| 	if (len != 8) {
 | |
| err_len:
 | |
| 		return("incorrect column length in SYS_COLUMNS");
 | |
| 	}
 | |
| 
 | |
| 	if (table_id) {
 | |
| 		*table_id = mach_read_from_8(field);
 | |
| 	} else if (table->id != mach_read_from_8(field)) {
 | |
| 		return dict_load_column_none;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__POS, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	pos = mach_read_from_4(field);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__DB_TRX_ID, &len);
 | |
| 	if (len != DATA_TRX_ID_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	const trx_id_t trx_id = trx_read_trx_id(field);
 | |
| 
 | |
| 	if (trx_id && mtr && use_uncommitted < 2
 | |
| 	    && trx_sys.find(nullptr, trx_id, false)) {
 | |
| 		if (use_uncommitted) {
 | |
| 			return dict_load_column_instant;
 | |
| 		}
 | |
| 		const auto savepoint = mtr->get_savepoint();
 | |
| 		dict_index_t* index = UT_LIST_GET_FIRST(
 | |
| 			dict_sys.sys_columns->indexes);
 | |
| 		rec_offs* offsets = rec_get_offsets(
 | |
| 			rec, index, nullptr, true, ULINT_UNDEFINED, &heap);
 | |
| 		const rec_t* old_vers;
 | |
| 		row_vers_build_for_semi_consistent_read(
 | |
| 			nullptr, rec, mtr, index, &offsets, &heap,
 | |
| 			heap, &old_vers, nullptr);
 | |
| 		mtr->rollback_to_savepoint(savepoint);
 | |
| 		rec = old_vers;
 | |
| 		if (!old_vers) {
 | |
| 			return dict_load_column_none;
 | |
| 		}
 | |
| 		ut_ad(!rec_get_deleted_flag(rec, 0));
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		ut_ad(trx_id);
 | |
| 		return dict_load_column_del;
 | |
| 	}
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__DB_ROLL_PTR, &len);
 | |
| 	if (len != DATA_ROLL_PTR_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__NAME, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	*col_name = name = mem_heap_strdupl(heap, (const char*) field, len);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__MTYPE, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	mtype = mach_read_from_4(field);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__PRTYPE, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	prtype = mach_read_from_4(field);
 | |
| 
 | |
| 	if (dtype_get_charset_coll(prtype) == 0
 | |
| 	    && dtype_is_string_type(mtype)) {
 | |
| 		/* The table was created with < 4.1.2. */
 | |
| 
 | |
| 		if (dtype_is_binary_string_type(mtype, prtype)) {
 | |
| 			/* Use the binary collation for
 | |
| 			string columns of binary type. */
 | |
| 
 | |
| 			prtype = dtype_form_prtype(
 | |
| 				prtype,
 | |
| 				DATA_MYSQL_BINARY_CHARSET_COLL);
 | |
| 		} else {
 | |
| 			/* Use the default charset for
 | |
| 			other than binary columns. */
 | |
| 
 | |
| 			prtype = dtype_form_prtype(
 | |
| 				prtype,
 | |
| 				data_mysql_default_charset_coll);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (table && table->n_def != pos && !(prtype & DATA_VIRTUAL)) {
 | |
| 		return("SYS_COLUMNS.POS mismatch");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__LEN, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	col_len = mach_read_from_4(field);
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_COLUMNS__PREC, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	num_base = mach_read_from_4(field);
 | |
| 
 | |
| 	if (table) {
 | |
| 		if (prtype & DATA_VIRTUAL) {
 | |
| #ifdef UNIV_DEBUG
 | |
| 			dict_v_col_t*	vcol =
 | |
| #endif
 | |
| 			dict_mem_table_add_v_col(
 | |
| 				table, heap, name, mtype,
 | |
| 				prtype, col_len,
 | |
| 				dict_get_v_col_mysql_pos(pos), num_base);
 | |
| 			ut_ad(vcol->v_pos == dict_get_v_col_pos(pos));
 | |
| 		} else {
 | |
| 			ut_ad(num_base == 0);
 | |
| 			dict_mem_table_add_col(table, heap, name, mtype,
 | |
| 					       prtype, col_len);
 | |
| 		}
 | |
| 
 | |
| 		if (trx_id > table->def_trx_id) {
 | |
| 			table->def_trx_id = trx_id;
 | |
| 		}
 | |
| 	} else {
 | |
| 		dict_mem_fill_column_struct(column, pos, mtype,
 | |
| 					    prtype, col_len);
 | |
| 	}
 | |
| 
 | |
| 	/* Report the virtual column number */
 | |
| 	if ((prtype & DATA_VIRTUAL) && nth_v_col != NULL) {
 | |
| 		*nth_v_col = dict_get_v_col_pos(pos);
 | |
| 	}
 | |
| 
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /** Error message for a delete-marked record in dict_load_virtual_low() */
 | |
| static const char *dict_load_virtual_del= "delete-marked record in SYS_VIRTUAL";
 | |
| static const char *dict_load_virtual_none= "SYS_VIRTUAL record not found";
 | |
| 
 | |
| /** Load a virtual column "mapping" (to base columns) information
 | |
| from a SYS_VIRTUAL record
 | |
| @param[in,out]	table		table
 | |
| @param[in]	uncommitted	false=READ COMMITTED, true=READ UNCOMMITTED
 | |
| @param[in,out]	column		mapped base column's dict_column_t
 | |
| @param[in,out]	table_id	table id
 | |
| @param[in,out]	pos		virtual column position
 | |
| @param[in,out]	base_pos	base column position
 | |
| @param[in]	rec		SYS_VIRTUAL record
 | |
| @return	error message
 | |
| @retval	NULL on success */
 | |
| static
 | |
| const char*
 | |
| dict_load_virtual_low(
 | |
| 	dict_table_t*	table,
 | |
| 	bool		uncommitted,
 | |
| 	dict_col_t**	column,
 | |
| 	table_id_t*	table_id,
 | |
| 	ulint*		pos,
 | |
| 	ulint*		base_pos,
 | |
| 	const rec_t*	rec)
 | |
| {
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 	ulint		base;
 | |
| 
 | |
| 	if (rec_get_n_fields_old(rec) != DICT_NUM_FIELDS__SYS_VIRTUAL) {
 | |
| 		return("wrong number of columns in SYS_VIRTUAL record");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_VIRTUAL__TABLE_ID, &len);
 | |
| 	if (len != 8) {
 | |
| err_len:
 | |
| 		return("incorrect column length in SYS_VIRTUAL");
 | |
| 	}
 | |
| 
 | |
| 	if (table_id != NULL) {
 | |
| 		*table_id = mach_read_from_8(field);
 | |
| 	} else if (table->id != mach_read_from_8(field)) {
 | |
| 		return dict_load_virtual_none;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_VIRTUAL__POS, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	if (pos != NULL) {
 | |
| 		*pos = mach_read_from_4(field);
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_VIRTUAL__BASE_POS, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	base = mach_read_from_4(field);
 | |
| 
 | |
| 	if (base_pos != NULL) {
 | |
| 		*base_pos = base;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_VIRTUAL__DB_TRX_ID, &len);
 | |
| 	if (len != DATA_TRX_ID_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_VIRTUAL__DB_ROLL_PTR, &len);
 | |
| 	if (len != DATA_ROLL_PTR_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	const trx_id_t trx_id = trx_read_trx_id(field);
 | |
| 
 | |
| 	if (trx_id && column && !uncommitted
 | |
| 	    && trx_sys.find(nullptr, trx_id, false)) {
 | |
| 		if (!rec_get_deleted_flag(rec, 0)) {
 | |
| 			return dict_load_virtual_none;
 | |
| 		}
 | |
| 	} else if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		ut_ad(trx_id != 0);
 | |
| 		return dict_load_virtual_del;
 | |
| 	}
 | |
| 
 | |
| 	if (column != NULL) {
 | |
| 		*column = dict_table_get_nth_col(table, base);
 | |
| 	}
 | |
| 
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /** Load the definitions for table columns.
 | |
| @param table           table
 | |
| @param use_uncommitted 0=READ COMMITTED, 1=detect, 2=READ UNCOMMITTED
 | |
| @param heap            memory heap for temporary storage
 | |
| @return error code
 | |
| @retval DB_SUCCESS on success
 | |
| @retval DB_SUCCESS_LOCKED_REC on success if use_uncommitted=1
 | |
| and instant ADD/DROP/reorder was detected */
 | |
| MY_ATTRIBUTE((nonnull, warn_unused_result))
 | |
| static dberr_t dict_load_columns(dict_table_t *table, unsigned use_uncommitted,
 | |
|                                  mem_heap_t *heap)
 | |
| {
 | |
| 	btr_pcur_t	pcur;
 | |
| 	mtr_t		mtr;
 | |
| 	ulint		n_skipped = 0;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	mtr.start();
 | |
| 
 | |
| 	dict_index_t* sys_index = dict_sys.sys_columns->indexes.start;
 | |
| 	ut_ad(!dict_sys.sys_columns->not_redundant());
 | |
| 
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_columns, sys_index,
 | |
| 			     DICT_FLD__SYS_COLUMNS__NAME, "NAME"));
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_columns, sys_index,
 | |
| 			     DICT_FLD__SYS_COLUMNS__PREC, "PREC"));
 | |
| 
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 	byte table_id[8];
 | |
| 	mach_write_to_8(table_id, table->id);
 | |
| 	dfield_set_data(&dfield, table_id, 8);
 | |
| 	dict_index_copy_types(&tuple, sys_index, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sys_index;
 | |
| 
 | |
| 	dberr_t err = btr_pcur_open_on_user_rec(&tuple,
 | |
| 						BTR_SEARCH_LEAF, &pcur, &mtr);
 | |
| 	if (err != DB_SUCCESS) {
 | |
| 		goto func_exit;
 | |
| 	}
 | |
| 
 | |
| 	ut_ad(table->n_t_cols == static_cast<ulint>(
 | |
| 	      table->n_cols) + static_cast<ulint>(table->n_v_cols));
 | |
| 
 | |
| 	for (ulint i = 0;
 | |
| 	     i + DATA_N_SYS_COLS < table->n_t_cols + n_skipped;
 | |
| 	     i++) {
 | |
| 		const char*	err_msg;
 | |
| 		const char*	name = NULL;
 | |
| 		ulint		nth_v_col = ULINT_UNDEFINED;
 | |
| 		const rec_t*	rec = btr_pcur_get_rec(&pcur);
 | |
| 
 | |
| 		err_msg = btr_pcur_is_on_user_rec(&pcur)
 | |
| 			? dict_load_column_low(table, use_uncommitted,
 | |
| 					       heap, NULL, NULL,
 | |
| 					       &name, rec, &mtr, &nth_v_col)
 | |
| 			: dict_load_column_none;
 | |
| 
 | |
| 		if (!err_msg) {
 | |
| 		} else if (err_msg == dict_load_column_del) {
 | |
| 			n_skipped++;
 | |
| 			goto next_rec;
 | |
| 		} else if (err_msg == dict_load_column_instant) {
 | |
| 			err = DB_SUCCESS_LOCKED_REC;
 | |
| 			goto func_exit;
 | |
| 		} else if (err_msg == dict_load_column_none
 | |
| 			   && strstr(table->name.m_name,
 | |
| 				     "/" TEMP_FILE_PREFIX_INNODB)) {
 | |
| 			break;
 | |
| 		} else {
 | |
| 			ib::error() << err_msg << " for table " << table->name;
 | |
| 			err = DB_CORRUPTION;
 | |
| 			goto func_exit;
 | |
| 		}
 | |
| 
 | |
| 		/* Note: Currently we have one DOC_ID column that is
 | |
| 		shared by all FTS indexes on a table. And only non-virtual
 | |
| 		column can be used for FULLTEXT index */
 | |
| 		if (innobase_strcasecmp(name,
 | |
| 					FTS_DOC_ID_COL_NAME) == 0
 | |
| 		    && nth_v_col == ULINT_UNDEFINED) {
 | |
| 			dict_col_t*	col;
 | |
| 			/* As part of normal loading of tables the
 | |
| 			table->flag is not set for tables with FTS
 | |
| 			till after the FTS indexes are loaded. So we
 | |
| 			create the fts_t instance here if there isn't
 | |
| 			one already created.
 | |
| 
 | |
| 			This case does not arise for table create as
 | |
| 			the flag is set before the table is created. */
 | |
| 			if (table->fts == NULL) {
 | |
| 				table->fts = fts_create(table);
 | |
| 				table->fts->cache = fts_cache_create(table);
 | |
| 				DICT_TF2_FLAG_SET(table, DICT_TF2_FTS_AUX_HEX_NAME);
 | |
| 			}
 | |
| 
 | |
| 			ut_a(table->fts->doc_col == ULINT_UNDEFINED);
 | |
| 
 | |
| 			col = dict_table_get_nth_col(table, i - n_skipped);
 | |
| 
 | |
| 			ut_ad(col->len == sizeof(doc_id_t));
 | |
| 
 | |
| 			if (col->prtype & DATA_FTS_DOC_ID) {
 | |
| 				DICT_TF2_FLAG_SET(
 | |
| 					table, DICT_TF2_FTS_HAS_DOC_ID);
 | |
| 				DICT_TF2_FLAG_UNSET(
 | |
| 					table, DICT_TF2_FTS_ADD_DOC_ID);
 | |
| 			}
 | |
| 
 | |
| 			table->fts->doc_col = i - n_skipped;
 | |
| 		}
 | |
| next_rec:
 | |
| 		btr_pcur_move_to_next_user_rec(&pcur, &mtr);
 | |
| 	}
 | |
| 
 | |
| func_exit:
 | |
| 	mtr.commit();
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| /** Loads SYS_VIRTUAL info for one virtual column
 | |
| @param table	   table definition
 | |
| @param uncommitted false=READ COMMITTED, true=READ UNCOMMITTED
 | |
| @param nth_v_col   virtual column position */
 | |
| MY_ATTRIBUTE((nonnull, warn_unused_result))
 | |
| static
 | |
| dberr_t
 | |
| dict_load_virtual_col(dict_table_t *table, bool uncommitted, ulint nth_v_col)
 | |
| {
 | |
| 	const dict_v_col_t* v_col = dict_table_get_nth_v_col(table, nth_v_col);
 | |
| 
 | |
| 	if (v_col->num_base == 0) {
 | |
| 		return DB_SUCCESS;
 | |
| 	}
 | |
| 
 | |
| 	dict_index_t*	sys_virtual_index;
 | |
| 	btr_pcur_t	pcur;
 | |
| 	mtr_t		mtr;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	mtr.start();
 | |
| 
 | |
| 	sys_virtual_index = dict_sys.sys_virtual->indexes.start;
 | |
| 	ut_ad(!dict_sys.sys_virtual->not_redundant());
 | |
| 
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_virtual, sys_virtual_index,
 | |
| 			     DICT_FLD__SYS_VIRTUAL__POS, "POS"));
 | |
| 
 | |
| 	dfield_t dfield[2];
 | |
| 	dtuple_t tuple{
 | |
| 		0,2,2,dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 	byte table_id[8], vcol_pos[4];
 | |
| 	mach_write_to_8(table_id, table->id);
 | |
| 	dfield_set_data(&dfield[0], table_id, 8);
 | |
| 	mach_write_to_4(vcol_pos,
 | |
| 			dict_create_v_col_pos(nth_v_col, v_col->m_col.ind));
 | |
| 	dfield_set_data(&dfield[1], vcol_pos, 4);
 | |
| 
 | |
| 	dict_index_copy_types(&tuple, sys_virtual_index, 2);
 | |
| 	pcur.btr_cur.page_cur.index = sys_virtual_index;
 | |
| 
 | |
| 	dberr_t err = btr_pcur_open_on_user_rec(&tuple,
 | |
| 						BTR_SEARCH_LEAF, &pcur, &mtr);
 | |
| 	if (err != DB_SUCCESS) {
 | |
| 		goto func_exit;
 | |
| 	}
 | |
| 
 | |
| 	for (ulint i = 0, skipped = 0;
 | |
| 	     i < unsigned{v_col->num_base} + skipped; i++) {
 | |
| 		ulint		pos;
 | |
| 		const char*	err_msg
 | |
| 			= btr_pcur_is_on_user_rec(&pcur)
 | |
| 			? dict_load_virtual_low(table, uncommitted,
 | |
| 						&v_col->base_col[i - skipped],
 | |
| 						NULL,
 | |
| 					        &pos, NULL,
 | |
| 						btr_pcur_get_rec(&pcur))
 | |
| 			: dict_load_virtual_none;
 | |
| 
 | |
| 		if (!err_msg) {
 | |
| 			ut_ad(pos == mach_read_from_4(vcol_pos));
 | |
| 		} else if (err_msg == dict_load_virtual_del) {
 | |
| 			skipped++;
 | |
| 		} else if (err_msg == dict_load_virtual_none
 | |
| 			   && strstr(table->name.m_name,
 | |
| 				     "/" TEMP_FILE_PREFIX_INNODB)) {
 | |
| 			break;
 | |
| 		} else {
 | |
| 			ib::error() << err_msg << " for table " << table->name;
 | |
| 			err = DB_CORRUPTION;
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		btr_pcur_move_to_next_user_rec(&pcur, &mtr);
 | |
| 	}
 | |
| 
 | |
| func_exit:
 | |
| 	mtr.commit();
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| /** Loads info from SYS_VIRTUAL for virtual columns.
 | |
| @param table	   table definition
 | |
| @param uncommitted false=READ COMMITTED, true=READ UNCOMMITTED */
 | |
| MY_ATTRIBUTE((nonnull, warn_unused_result))
 | |
| static dberr_t dict_load_virtual(dict_table_t *table, bool uncommitted)
 | |
| {
 | |
|   for (ulint i= 0; i < table->n_v_cols; i++)
 | |
|     if (dberr_t err= dict_load_virtual_col(table, uncommitted, i))
 | |
|       return err;
 | |
|   return DB_SUCCESS;
 | |
| }
 | |
| 
 | |
| /** Error message for a delete-marked record in dict_load_field_low() */
 | |
| static const char *dict_load_field_del= "delete-marked record in SYS_FIELDS";
 | |
| 
 | |
| static const char *dict_load_field_none= "SYS_FIELDS record not found";
 | |
| 
 | |
| /** Load an index field definition from a SYS_FIELDS record to dict_index_t.
 | |
| @return	error message
 | |
| @retval	NULL on success */
 | |
| static
 | |
| const char*
 | |
| dict_load_field_low(
 | |
| 	byte*		index_id,	/*!< in/out: index id (8 bytes)
 | |
| 					an "in" value if index != NULL
 | |
| 					and "out" if index == NULL */
 | |
| 	bool		uncommitted,	/*!< in: false=READ COMMITTED,
 | |
| 					true=READ UNCOMMITTED */
 | |
| 	dict_index_t*	index,		/*!< in/out: index, could be NULL
 | |
| 					if we just populate a dict_field_t
 | |
| 					struct with information from
 | |
| 					a SYS_FIELDS record */
 | |
| 	dict_field_t*	sys_field,	/*!< out: dict_field_t to be
 | |
| 					filled */
 | |
| 	ulint*		pos,		/*!< out: Field position */
 | |
| 	byte*		last_index_id,	/*!< in: last index id */
 | |
| 	mem_heap_t*	heap,		/*!< in/out: memory heap
 | |
| 					for temporary storage */
 | |
| 	mtr_t*		mtr,		/*!< in/out: mini-transaction */
 | |
| 	const rec_t*	rec)		/*!< in: SYS_FIELDS record */
 | |
| {
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 	unsigned	pos_and_prefix_len;
 | |
| 	unsigned	prefix_len;
 | |
| 	bool		descending;
 | |
| 	bool		first_field;
 | |
| 	ulint		position;
 | |
| 
 | |
| 	/* Either index or sys_field is supplied, not both */
 | |
| 	ut_ad((!index) != (!sys_field));
 | |
| 	ut_ad((!index) == !mtr);
 | |
| 
 | |
| 	if (rec_get_n_fields_old(rec) != DICT_NUM_FIELDS__SYS_FIELDS) {
 | |
| 		return("wrong number of columns in SYS_FIELDS record");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FIELDS__INDEX_ID, &len);
 | |
| 	if (len != 8) {
 | |
| err_len:
 | |
| 		return("incorrect column length in SYS_FIELDS");
 | |
| 	}
 | |
| 
 | |
| 	if (!index) {
 | |
| 		ut_a(last_index_id);
 | |
| 		memcpy(index_id, (const char*) field, 8);
 | |
| 		first_field = memcmp(index_id, last_index_id, 8);
 | |
| 	} else {
 | |
| 		first_field = (index->n_def == 0);
 | |
| 		if (memcmp(field, index_id, 8)) {
 | |
| 			return dict_load_field_none;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* The next field stores the field position in the index and a
 | |
| 	possible column prefix length if the index field does not
 | |
| 	contain the whole column. The storage format is like this: if
 | |
| 	there is at least one prefix field in the index, then the HIGH
 | |
| 	2 bytes contain the field number (index->n_def) and the low 2
 | |
| 	bytes the prefix length for the field. Otherwise the field
 | |
| 	number (index->n_def) is contained in the 2 LOW bytes. */
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FIELDS__POS, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	pos_and_prefix_len = mach_read_from_4(field);
 | |
| 
 | |
| 	if (index && UNIV_UNLIKELY
 | |
| 	    ((pos_and_prefix_len & 0xFFFFUL) != index->n_def
 | |
| 	     && (pos_and_prefix_len >> 16 & 0xFFFF) != index->n_def)) {
 | |
| 		return("SYS_FIELDS.POS mismatch");
 | |
| 	}
 | |
| 
 | |
| 	if (first_field || pos_and_prefix_len > 0xFFFFUL) {
 | |
| 		prefix_len = pos_and_prefix_len & 0x7FFFUL;
 | |
| 		descending = (pos_and_prefix_len & 0x8000UL);
 | |
| 		position = (pos_and_prefix_len & 0xFFFF0000UL)  >> 16;
 | |
| 	} else {
 | |
| 		prefix_len = 0;
 | |
| 		descending = false;
 | |
| 		position = pos_and_prefix_len & 0xFFFFUL;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FIELDS__DB_TRX_ID, &len);
 | |
| 	if (len != DATA_TRX_ID_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_FIELDS__DB_ROLL_PTR, &len);
 | |
| 	if (len != DATA_ROLL_PTR_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	const trx_id_t trx_id = trx_read_trx_id(field);
 | |
| 
 | |
| 	if (!trx_id) {
 | |
| 		ut_ad(!rec_get_deleted_flag(rec, 0));
 | |
| 	} else if (!mtr || uncommitted) {
 | |
| 	} else if (trx_sys.find(nullptr, trx_id, false)) {
 | |
| 		const auto savepoint = mtr->get_savepoint();
 | |
| 		dict_index_t* sys_field = UT_LIST_GET_FIRST(
 | |
| 			dict_sys.sys_fields->indexes);
 | |
| 		rec_offs* offsets = rec_get_offsets(
 | |
| 			rec, sys_field, nullptr, true, ULINT_UNDEFINED, &heap);
 | |
| 		const rec_t* old_vers;
 | |
| 		row_vers_build_for_semi_consistent_read(
 | |
| 			nullptr, rec, mtr, sys_field, &offsets, &heap,
 | |
| 			heap, &old_vers, nullptr);
 | |
| 		mtr->rollback_to_savepoint(savepoint);
 | |
| 		rec = old_vers;
 | |
| 		if (!old_vers || rec_get_deleted_flag(rec, 0)) {
 | |
| 			return dict_load_field_none;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		return(dict_load_field_del);
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FIELDS__COL_NAME, &len);
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	if (index) {
 | |
| 		dict_mem_index_add_field(
 | |
| 			index, mem_heap_strdupl(heap, (const char*) field, len),
 | |
| 			prefix_len, descending);
 | |
| 	} else {
 | |
| 		sys_field->name = mem_heap_strdupl(
 | |
| 			heap, (const char*) field, len);
 | |
| 		sys_field->prefix_len = prefix_len & ((1U << 12) - 1);
 | |
| 		sys_field->descending = descending;
 | |
| 		*pos = position;
 | |
| 	}
 | |
| 
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /**
 | |
| Load definitions for index fields.
 | |
| @param index       index whose fields are to be loaded
 | |
| @param uncommitted false=READ COMMITTED, true=READ UNCOMMITTED
 | |
| @param heap        memory heap for temporary storage
 | |
| @return error code
 | |
| @return DB_SUCCESS if the fields were loaded successfully */
 | |
| static dberr_t dict_load_fields(dict_index_t *index, bool uncommitted,
 | |
|                                 mem_heap_t *heap)
 | |
| {
 | |
| 	btr_pcur_t	pcur;
 | |
| 	mtr_t		mtr;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	mtr.start();
 | |
| 
 | |
| 	dict_index_t* sys_index = dict_sys.sys_fields->indexes.start;
 | |
| 	ut_ad(!dict_sys.sys_fields->not_redundant());
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_fields, sys_index,
 | |
| 			     DICT_FLD__SYS_FIELDS__COL_NAME, "COL_NAME"));
 | |
| 
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 	byte index_id[8];
 | |
| 	mach_write_to_8(index_id, index->id);
 | |
| 	dfield_set_data(&dfield, index_id, 8);
 | |
| 	dict_index_copy_types(&tuple, sys_index, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sys_index;
 | |
| 
 | |
| 	dberr_t error = btr_pcur_open_on_user_rec(&tuple, BTR_SEARCH_LEAF,
 | |
| 						  &pcur, &mtr);
 | |
| 	if (error != DB_SUCCESS) {
 | |
| 		goto func_exit;
 | |
| 	}
 | |
| 
 | |
| 	for (ulint i = 0; i < index->n_fields; i++) {
 | |
| 		const char *err_msg = btr_pcur_is_on_user_rec(&pcur)
 | |
| 			? dict_load_field_low(index_id, uncommitted, index,
 | |
| 					      nullptr, nullptr, nullptr,
 | |
| 					      heap, &mtr,
 | |
| 					      btr_pcur_get_rec(&pcur))
 | |
| 			: dict_load_field_none;
 | |
| 
 | |
| 		if (!err_msg) {
 | |
| 		} else if (err_msg == dict_load_field_del) {
 | |
| 			/* There could be delete marked records in
 | |
| 			SYS_FIELDS because SYS_FIELDS.INDEX_ID can be
 | |
| 			updated by ALTER TABLE ADD INDEX. */
 | |
| 		} else {
 | |
| 			if (err_msg != dict_load_field_none
 | |
| 			    || strstr(index->table->name.m_name,
 | |
| 				      "/" TEMP_FILE_PREFIX_INNODB)) {
 | |
| 				ib::error() << err_msg << " for index "
 | |
| 					    << index->name
 | |
| 					    << " of table "
 | |
| 					    << index->table->name;
 | |
| 			}
 | |
| 			error = DB_CORRUPTION;
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		btr_pcur_move_to_next_user_rec(&pcur, &mtr);
 | |
| 	}
 | |
| 
 | |
| func_exit:
 | |
| 	mtr.commit();
 | |
| 	return error;
 | |
| }
 | |
| 
 | |
| /** Error message for a delete-marked record in dict_load_index_low() */
 | |
| static const char *dict_load_index_del= "delete-marked record in SYS_INDEXES";
 | |
| /** Error message for table->id mismatch in dict_load_index_low() */
 | |
| static const char *dict_load_index_none= "SYS_INDEXES record not found";
 | |
| /** Error message for SYS_TABLES flags mismatch in dict_load_table_low() */
 | |
| static const char *dict_load_table_flags= "incorrect flags in SYS_TABLES";
 | |
| 
 | |
| /** Load an index definition from a SYS_INDEXES record to dict_index_t.
 | |
| @return	error message
 | |
| @retval	NULL on success */
 | |
| static
 | |
| const char*
 | |
| dict_load_index_low(
 | |
| 	byte*		table_id,	/*!< in/out: table id (8 bytes),
 | |
| 					an "in" value if mtr
 | |
| 					and "out" when !mtr */
 | |
| 	bool		uncommitted,	/*!< in: false=READ COMMITTED,
 | |
| 					true=READ UNCOMMITTED */
 | |
| 	mem_heap_t*	heap,		/*!< in/out: temporary memory heap */
 | |
| 	const rec_t*	rec,		/*!< in: SYS_INDEXES record */
 | |
| 	mtr_t*		mtr,		/*!< in/out: mini-transaction,
 | |
| 					or nullptr if a pre-allocated
 | |
| 					*index is to be filled in */
 | |
| 	dict_table_t*	table,		/*!< in/out: table, or NULL */
 | |
| 	dict_index_t**	index)		/*!< out,own: index, or NULL */
 | |
| {
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 	index_id_t	id;
 | |
| 	ulint		n_fields;
 | |
| 	ulint		type;
 | |
| 	unsigned	merge_threshold;
 | |
| 
 | |
| 	if (mtr) {
 | |
| 		*index = NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_n_fields_old(rec) == DICT_NUM_FIELDS__SYS_INDEXES) {
 | |
| 		/* MERGE_THRESHOLD exists */
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_INDEXES__MERGE_THRESHOLD, &len);
 | |
| 		switch (len) {
 | |
| 		case 4:
 | |
| 			merge_threshold = mach_read_from_4(field);
 | |
| 			break;
 | |
| 		case UNIV_SQL_NULL:
 | |
| 			merge_threshold = DICT_INDEX_MERGE_THRESHOLD_DEFAULT;
 | |
| 			break;
 | |
| 		default:
 | |
| 			return("incorrect MERGE_THRESHOLD length"
 | |
| 			       " in SYS_INDEXES");
 | |
| 		}
 | |
| 	} else if (rec_get_n_fields_old(rec)
 | |
| 		   == DICT_NUM_FIELDS__SYS_INDEXES - 1) {
 | |
| 		/* MERGE_THRESHOLD doesn't exist */
 | |
| 
 | |
| 		merge_threshold = DICT_INDEX_MERGE_THRESHOLD_DEFAULT;
 | |
| 	} else {
 | |
| 		return("wrong number of columns in SYS_INDEXES record");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_INDEXES__TABLE_ID, &len);
 | |
| 	if (len != 8) {
 | |
| err_len:
 | |
| 		return("incorrect column length in SYS_INDEXES");
 | |
| 	}
 | |
| 
 | |
| 	if (!mtr) {
 | |
| 		/* We are reading a SYS_INDEXES record. Copy the table_id */
 | |
| 		memcpy(table_id, (const char*) field, 8);
 | |
| 	} else if (memcmp(field, table_id, 8)) {
 | |
| 		/* Caller supplied table_id, verify it is the same
 | |
| 		id as on the index record */
 | |
| 		return dict_load_index_none;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_INDEXES__ID, &len);
 | |
| 	if (len != 8) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	id = mach_read_from_8(field);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_INDEXES__DB_TRX_ID, &len);
 | |
| 	if (len != DATA_TRX_ID_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	rec_get_nth_field_offs_old(
 | |
| 		rec, DICT_FLD__SYS_INDEXES__DB_ROLL_PTR, &len);
 | |
| 	if (len != DATA_ROLL_PTR_LEN && len != UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	const trx_id_t trx_id = trx_read_trx_id(field);
 | |
| 	if (!trx_id) {
 | |
| 		ut_ad(!rec_get_deleted_flag(rec, 0));
 | |
| 	} else if (!mtr || uncommitted) {
 | |
| 	} else if (trx_sys.find(nullptr, trx_id, false)) {
 | |
| 		const auto savepoint = mtr->get_savepoint();
 | |
| 		dict_index_t* sys_index = UT_LIST_GET_FIRST(
 | |
| 			dict_sys.sys_indexes->indexes);
 | |
| 		rec_offs* offsets = rec_get_offsets(
 | |
| 			rec, sys_index, nullptr, true, ULINT_UNDEFINED, &heap);
 | |
| 		const rec_t* old_vers;
 | |
| 		row_vers_build_for_semi_consistent_read(
 | |
| 			nullptr, rec, mtr, sys_index, &offsets, &heap,
 | |
| 			heap, &old_vers, nullptr);
 | |
| 		mtr->rollback_to_savepoint(savepoint);
 | |
| 		rec = old_vers;
 | |
| 		if (!old_vers || rec_get_deleted_flag(rec, 0)) {
 | |
| 			return dict_load_index_none;
 | |
| 		}
 | |
| 	} else if (rec_get_deleted_flag(rec, 0)
 | |
| 		   && rec[8 + 8 + DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN]
 | |
| 		   != static_cast<byte>(*TEMP_INDEX_PREFIX_STR)
 | |
| 		   && table->def_trx_id < trx_id) {
 | |
| 		table->def_trx_id = trx_id;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_INDEXES__N_FIELDS, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	n_fields = mach_read_from_4(field);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_INDEXES__TYPE, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 	type = mach_read_from_4(field);
 | |
| 	if (type & (~0U << DICT_IT_BITS)) {
 | |
| 		return("unknown SYS_INDEXES.TYPE bits");
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_INDEXES__PAGE_NO, &len);
 | |
| 	if (len != 4) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	ut_d(const auto name_offs =)
 | |
| 	rec_get_nth_field_offs_old(rec, DICT_FLD__SYS_INDEXES__NAME, &len);
 | |
| 	ut_ad(name_offs == 8 + 8 + DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN);
 | |
| 
 | |
| 	if (len == 0 || len == UNIV_SQL_NULL) {
 | |
| 		goto err_len;
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		return dict_load_index_del;
 | |
| 	}
 | |
| 
 | |
| 	char* name = mem_heap_strdupl(heap, reinterpret_cast<const char*>(rec)
 | |
| 				      + (8 + 8 + DATA_TRX_ID_LEN
 | |
| 					 + DATA_ROLL_PTR_LEN),
 | |
| 				      len);
 | |
| 
 | |
| 	if (mtr) {
 | |
| 		*index = dict_mem_index_create(table, name, type, n_fields);
 | |
| 	} else {
 | |
| 		dict_mem_fill_index_struct(*index, nullptr, name,
 | |
| 					   type, n_fields);
 | |
| 	}
 | |
| 
 | |
| 	(*index)->id = id;
 | |
| 	(*index)->page = mach_read_from_4(field);
 | |
| 	ut_ad((*index)->page);
 | |
| 	(*index)->merge_threshold = merge_threshold & ((1U << 6) - 1);
 | |
| 
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /** Load definitions for table indexes. Adds them to the data dictionary cache.
 | |
| @param table       table definition
 | |
| @param uncommitted false=READ COMMITTED, true=READ UNCOMMITTED
 | |
| @param heap        memory heap for temporary storage
 | |
| @param ignore_err  errors to be ignored when loading the index definition
 | |
| @return error code
 | |
| @retval DB_SUCCESS if all indexes were successfully loaded
 | |
| @retval DB_CORRUPTION if corruption of dictionary table
 | |
| @retval DB_UNSUPPORTED if table has unknown index type */
 | |
| static MY_ATTRIBUTE((nonnull))
 | |
| dberr_t dict_load_indexes(dict_table_t *table, bool uncommitted,
 | |
|                           mem_heap_t *heap, dict_err_ignore_t ignore_err)
 | |
| {
 | |
| 	dict_index_t*	sys_index;
 | |
| 	btr_pcur_t	pcur;
 | |
| 	byte		table_id[8];
 | |
| 	mtr_t		mtr;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	mtr.start();
 | |
| 
 | |
| 	sys_index = dict_sys.sys_indexes->indexes.start;
 | |
| 	ut_ad(!dict_sys.sys_indexes->not_redundant());
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_indexes, sys_index,
 | |
| 			     DICT_FLD__SYS_INDEXES__NAME, "NAME"));
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_indexes, sys_index,
 | |
| 			     DICT_FLD__SYS_INDEXES__PAGE_NO, "PAGE_NO"));
 | |
| 
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 	mach_write_to_8(table_id, table->id);
 | |
| 	dfield_set_data(&dfield, table_id, 8);
 | |
| 	dict_index_copy_types(&tuple, sys_index, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sys_index;
 | |
| 
 | |
| 	dberr_t error = btr_pcur_open_on_user_rec(&tuple, BTR_SEARCH_LEAF,
 | |
| 						  &pcur, &mtr);
 | |
| 	if (error != DB_SUCCESS) {
 | |
| 		goto func_exit;
 | |
| 	}
 | |
| 
 | |
| 	while (btr_pcur_is_on_user_rec(&pcur)) {
 | |
| 		dict_index_t*	index = NULL;
 | |
| 		const char*	err_msg;
 | |
| 		const rec_t*	rec = btr_pcur_get_rec(&pcur);
 | |
| 		if ((ignore_err & DICT_ERR_IGNORE_RECOVER_LOCK)
 | |
| 		    && (rec_get_n_fields_old(rec)
 | |
| 			== DICT_NUM_FIELDS__SYS_INDEXES
 | |
| 			/* a record for older SYS_INDEXES table
 | |
| 			(missing merge_threshold column) is acceptable. */
 | |
| 			|| rec_get_n_fields_old(rec)
 | |
| 			   == DICT_NUM_FIELDS__SYS_INDEXES - 1)) {
 | |
| 			const byte*	field;
 | |
| 			ulint		len;
 | |
| 			field = rec_get_nth_field_old(
 | |
| 				rec, DICT_FLD__SYS_INDEXES__NAME, &len);
 | |
| 
 | |
| 			if (len != UNIV_SQL_NULL
 | |
| 			    && static_cast<char>(*field)
 | |
| 			    == static_cast<char>(*TEMP_INDEX_PREFIX_STR)) {
 | |
| 				/* Skip indexes whose name starts with
 | |
| 				TEMP_INDEX_PREFIX_STR, because they will
 | |
| 				be dropped by row_merge_drop_temp_indexes()
 | |
| 				during crash recovery. */
 | |
| 				goto next_rec;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		err_msg = dict_load_index_low(table_id, uncommitted, heap, rec,
 | |
| 					      &mtr, table, &index);
 | |
| 		ut_ad(!index == !!err_msg);
 | |
| 
 | |
| 		if (err_msg == dict_load_index_none) {
 | |
| 			/* We have ran out of index definitions for
 | |
| 			the table. */
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		if (err_msg == dict_load_index_del) {
 | |
| 			goto next_rec;
 | |
| 		} else if (err_msg) {
 | |
| 			ib::error() << err_msg;
 | |
| 			if (ignore_err & DICT_ERR_IGNORE_INDEX) {
 | |
| 				goto next_rec;
 | |
| 			}
 | |
| 			error = DB_CORRUPTION;
 | |
| 			goto func_exit;
 | |
| 		} else if (rec[8 + 8 + DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN]
 | |
| 			   == static_cast<byte>(*TEMP_INDEX_PREFIX_STR)) {
 | |
| 			dict_mem_index_free(index);
 | |
| 			goto next_rec;
 | |
| 		} else {
 | |
| 			const trx_id_t id = trx_read_trx_id(rec + 8 + 8);
 | |
| 			if (id > table->def_trx_id) {
 | |
| 				table->def_trx_id = id;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		ut_ad(index);
 | |
| 		ut_ad(!dict_index_is_online_ddl(index));
 | |
| 
 | |
| 		/* Check whether the index is corrupted */
 | |
| 		if (ignore_err != DICT_ERR_IGNORE_DROP
 | |
| 		    && index->is_corrupted() && index->is_clust()) {
 | |
| 			dict_mem_index_free(index);
 | |
| 			error = DB_TABLE_CORRUPT;
 | |
| 			goto func_exit;
 | |
| 		}
 | |
| 
 | |
| 		if (index->type & DICT_FTS
 | |
| 		    && !dict_table_has_fts_index(table)) {
 | |
| 			/* This should have been created by now. */
 | |
| 			ut_a(table->fts != NULL);
 | |
| 			DICT_TF2_FLAG_SET(table, DICT_TF2_FTS);
 | |
| 		}
 | |
| 
 | |
| 		/* We check for unsupported types first, so that the
 | |
| 		subsequent checks are relevant for the supported types. */
 | |
| 		if (index->type & ~(DICT_CLUSTERED | DICT_UNIQUE
 | |
| 				    | DICT_CORRUPT | DICT_FTS
 | |
| 				    | DICT_SPATIAL | DICT_VIRTUAL)) {
 | |
| 
 | |
| 			ib::error() << "Unknown type " << index->type
 | |
| 				<< " of index " << index->name
 | |
| 				<< " of table " << table->name;
 | |
| 
 | |
| 			error = DB_UNSUPPORTED;
 | |
| 			dict_mem_index_free(index);
 | |
| 			goto func_exit;
 | |
| 		} else if (index->page == FIL_NULL
 | |
| 			   && table->is_readable()
 | |
| 			   && (!(index->type & DICT_FTS))) {
 | |
| 			if (!uncommitted
 | |
| 			    && ignore_err != DICT_ERR_IGNORE_DROP) {
 | |
| 				ib::error_or_warn(!(ignore_err
 | |
| 						    & DICT_ERR_IGNORE_INDEX))
 | |
| 					<< "Index " << index->name
 | |
| 					<< " for table " << table->name
 | |
| 					<< " has been freed!";
 | |
| 			}
 | |
| 
 | |
| 			if (!(ignore_err & DICT_ERR_IGNORE_INDEX)) {
 | |
| corrupted:
 | |
| 				dict_mem_index_free(index);
 | |
| 				error = DB_CORRUPTION;
 | |
| 				goto func_exit;
 | |
| 			}
 | |
| 			/* If caller can tolerate this error,
 | |
| 			we will continue to load the index and
 | |
| 			let caller deal with this error. However
 | |
| 			mark the index and table corrupted. We
 | |
| 			only need to mark such in the index
 | |
| 			dictionary cache for such metadata corruption,
 | |
| 			since we would always be able to set it
 | |
| 			when loading the dictionary cache */
 | |
| 			if (index->is_clust()) {
 | |
| 				index->table->corrupted = true;
 | |
| 				index->table->file_unreadable = true;
 | |
| 			}
 | |
| 			index->type |= DICT_CORRUPT;
 | |
| 		} else if (!dict_index_is_clust(index)
 | |
| 			   && NULL == dict_table_get_first_index(table)) {
 | |
| 
 | |
| 			ib::error() << "Trying to load index " << index->name
 | |
| 				<< " for table " << table->name
 | |
| 				<< ", but the first index is not clustered!";
 | |
| 
 | |
| 			goto corrupted;
 | |
| 		} else if (dict_is_sys_table(table->id)
 | |
| 			   && (dict_index_is_clust(index)
 | |
| 			       || ((table == dict_sys.sys_tables)
 | |
| 				   && !strcmp("ID_IND", index->name)))) {
 | |
| 
 | |
| 			/* The index was created in memory already at booting
 | |
| 			of the database server */
 | |
| 			dict_mem_index_free(index);
 | |
| 		} else {
 | |
| 			error = dict_load_fields(index, uncommitted, heap);
 | |
| 			if (error != DB_SUCCESS) {
 | |
| 				goto func_exit;
 | |
| 			}
 | |
| 
 | |
| 			/* The data dictionary tables should never contain
 | |
| 			invalid index definitions.  If we ignored this error
 | |
| 			and simply did not load this index definition, the
 | |
| 			.frm file would disagree with the index definitions
 | |
| 			inside InnoDB. */
 | |
| 			if ((error = dict_index_add_to_cache(index,
 | |
| 							     index->page))
 | |
| 			    != DB_SUCCESS) {
 | |
| 				goto func_exit;
 | |
| 			}
 | |
| 
 | |
| #ifdef UNIV_DEBUG
 | |
| 			// The following assertion doesn't hold for FTS indexes
 | |
| 			// as it may have prefix_len=1 with any charset
 | |
| 			if (index->type != DICT_FTS) {
 | |
| 				for (uint i = 0; i < index->n_fields; i++) {
 | |
| 					dict_field_t &f = index->fields[i];
 | |
| 					ut_ad(f.col->mbmaxlen == 0
 | |
| 					      || f.prefix_len
 | |
| 					      % f.col->mbmaxlen == 0);
 | |
| 				}
 | |
| 			}
 | |
| #endif /* UNIV_DEBUG */
 | |
| 		}
 | |
| next_rec:
 | |
| 		btr_pcur_move_to_next_user_rec(&pcur, &mtr);
 | |
| 	}
 | |
| 
 | |
| 	if (!dict_table_get_first_index(table)
 | |
| 	    && !(ignore_err & DICT_ERR_IGNORE_INDEX)) {
 | |
| 		ib::warn() << "No indexes found for table " << table->name;
 | |
| 		error = DB_CORRUPTION;
 | |
| 		goto func_exit;
 | |
| 	}
 | |
| 
 | |
| 	ut_ad(table->fts_doc_id_index == NULL);
 | |
| 
 | |
| 	if (table->fts != NULL) {
 | |
| 		dict_index_t *idx = dict_table_get_index_on_name(
 | |
| 			table, FTS_DOC_ID_INDEX_NAME);
 | |
| 		if (idx && dict_index_is_unique(idx)) {
 | |
| 			table->fts_doc_id_index = idx;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* If the table contains FTS indexes, populate table->fts->indexes */
 | |
| 	if (dict_table_has_fts_index(table)) {
 | |
| 		ut_ad(table->fts_doc_id_index != NULL);
 | |
| 		/* table->fts->indexes should have been created. */
 | |
| 		ut_a(table->fts->indexes != NULL);
 | |
| 		dict_table_get_all_fts_indexes(table, table->fts->indexes);
 | |
| 	}
 | |
| 
 | |
| func_exit:
 | |
| 	mtr.commit();
 | |
| 	return error;
 | |
| }
 | |
| 
 | |
| /** Load a table definition from a SYS_TABLES record to dict_table_t.
 | |
| Do not load any columns or indexes.
 | |
| @param[in,out]	mtr		mini-transaction
 | |
| @param[in]	uncommitted	whether to use READ UNCOMMITTED isolation level
 | |
| @param[in]	rec		SYS_TABLES record
 | |
| @param[out,own]	table		table, or nullptr
 | |
| @return	error message
 | |
| @retval	nullptr on success */
 | |
| const char *dict_load_table_low(mtr_t *mtr, bool uncommitted,
 | |
|                                 const rec_t *rec, dict_table_t **table)
 | |
| {
 | |
| 	table_id_t	table_id;
 | |
| 	uint32_t	space_id, t_num, flags, flags2;
 | |
| 	ulint		n_cols, n_v_col;
 | |
| 	trx_id_t	trx_id;
 | |
| 
 | |
| 	if (const char* error_text = dict_sys_tables_rec_check(rec)) {
 | |
| 		*table = NULL;
 | |
| 		return(error_text);
 | |
| 	}
 | |
| 
 | |
| 	if (auto r = dict_sys_tables_rec_read(rec, uncommitted, mtr,
 | |
| 					      &table_id, &space_id,
 | |
| 					      &t_num, &flags, &flags2,
 | |
| 					      &trx_id)) {
 | |
| 		*table = NULL;
 | |
| 		return r == READ_ERROR ? dict_load_table_flags : nullptr;
 | |
| 	}
 | |
| 
 | |
| 	dict_table_decode_n_col(t_num, &n_cols, &n_v_col);
 | |
| 
 | |
| 	*table = dict_table_t::create(
 | |
| 		span<const char>(reinterpret_cast<const char*>(rec),
 | |
| 				 rec_get_field_start_offs(rec, 1)),
 | |
| 		nullptr, n_cols + n_v_col, n_v_col, flags, flags2);
 | |
| 	(*table)->space_id = space_id;
 | |
| 	(*table)->id = table_id;
 | |
| 	(*table)->file_unreadable = !!(flags2 & DICT_TF2_DISCARDED);
 | |
| 	(*table)->def_trx_id = trx_id;
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /** Make sure the data_file_name is saved in dict_table_t if needed.
 | |
| @param[in,out]	table		Table object */
 | |
| void dict_get_and_save_data_dir_path(dict_table_t *table)
 | |
| {
 | |
|   ut_ad(!table->is_temporary());
 | |
|   ut_ad(!table->space || table->space->id == table->space_id);
 | |
| 
 | |
|   if (!table->data_dir_path && table->space_id && table->space)
 | |
|   {
 | |
|     const char *filepath= table->space->chain.start->name;
 | |
|     if (strncmp(fil_path_to_mysql_datadir, filepath,
 | |
|                 strlen(fil_path_to_mysql_datadir)))
 | |
|     {
 | |
|       table->lock_mutex_lock();
 | |
|       table->flags|= 1 << DICT_TF_POS_DATA_DIR & ((1U << DICT_TF_BITS) - 1);
 | |
|       table->data_dir_path= mem_heap_strdup(table->heap, filepath);
 | |
|       os_file_make_data_dir_path(table->data_dir_path);
 | |
|       table->lock_mutex_unlock();
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /** Opens a tablespace for dict_load_table_one()
 | |
| @param[in,out]	table		A table that refers to the tablespace to open
 | |
| @param[in]	ignore_err	Whether to ignore an error. */
 | |
| UNIV_INLINE
 | |
| void
 | |
| dict_load_tablespace(
 | |
| 	dict_table_t*		table,
 | |
| 	dict_err_ignore_t	ignore_err)
 | |
| {
 | |
| 	ut_ad(!table->is_temporary());
 | |
| 	ut_ad(!table->space);
 | |
| 	ut_ad(table->space_id < SRV_SPACE_ID_UPPER_BOUND);
 | |
| 	ut_ad(fil_system.sys_space);
 | |
| 
 | |
| 	if (table->space_id == TRX_SYS_SPACE) {
 | |
| 		table->space = fil_system.sys_space;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (table->flags2 & DICT_TF2_DISCARDED) {
 | |
| 		ib::warn() << "Tablespace for table " << table->name
 | |
| 			<< " is set as discarded.";
 | |
| 		table->file_unreadable = true;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* The tablespace may already be open. */
 | |
| 	table->space = fil_space_for_table_exists_in_mem(table->space_id,
 | |
| 							 table->flags);
 | |
| 	if (table->space || table->file_unreadable) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* Use the remote filepath if needed. This parameter is optional
 | |
| 	in the call to fil_ibd_open(). If not supplied, it will be built
 | |
| 	from the table->name. */
 | |
| 	char* filepath = NULL;
 | |
| 	if (DICT_TF_HAS_DATA_DIR(table->flags)) {
 | |
| 		/* This will set table->data_dir_path from fil_system */
 | |
| 		dict_get_and_save_data_dir_path(table);
 | |
| 
 | |
| 		if (table->data_dir_path) {
 | |
| 			filepath = fil_make_filepath(
 | |
| 				table->data_dir_path, table->name, IBD, true);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	table->space = fil_ibd_open(
 | |
| 		table->space_id, dict_tf_to_fsp_flags(table->flags),
 | |
| 		fil_space_t::VALIDATE_SPACE_ID,
 | |
| 		{table->name.m_name, strlen(table->name.m_name)}, filepath);
 | |
| 
 | |
| 	if (!table->space) {
 | |
| 		/* We failed to find a sensible tablespace file */
 | |
| 		table->file_unreadable = true;
 | |
| 
 | |
| 		if (!(ignore_err & DICT_ERR_IGNORE_RECOVER_LOCK)) {
 | |
| 			sql_print_error("InnoDB: Failed to load tablespace %"
 | |
| 					PRIu32 " for table %s",
 | |
| 					table->space_id, table->name.m_name);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ut_free(filepath);
 | |
| }
 | |
| 
 | |
| /** Loads a table definition and also all its index definitions.
 | |
| 
 | |
| Loads those foreign key constraints whose referenced table is already in
 | |
| dictionary cache.  If a foreign key constraint is not loaded, then the
 | |
| referenced table is pushed into the output stack (fk_tables), if it is not
 | |
| NULL.  These tables must be subsequently loaded so that all the foreign
 | |
| key constraints are loaded into memory.
 | |
| 
 | |
| @param[in]	name		Table name in the db/tablename format
 | |
| @param[in]	ignore_err	Error to be ignored when loading table
 | |
| 				and its index definition
 | |
| @param[out]	fk_tables	Related table names that must also be
 | |
| 				loaded to ensure that all foreign key
 | |
| 				constraints are loaded.
 | |
| @return table, possibly with file_unreadable flag set
 | |
| @retval nullptr if the table does not exist */
 | |
| static dict_table_t *dict_load_table_one(const span<const char> &name,
 | |
|                                          dict_err_ignore_t ignore_err,
 | |
|                                          dict_names_t &fk_tables)
 | |
| {
 | |
| 	btr_pcur_t	pcur;
 | |
| 	mtr_t		mtr;
 | |
| 
 | |
| 	DBUG_ENTER("dict_load_table_one");
 | |
| 	DBUG_PRINT("dict_load_table_one",
 | |
| 		   ("table: %.*s", int(name.size()), name.data()));
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	dict_index_t *sys_index = dict_sys.sys_tables->indexes.start;
 | |
| 	ut_ad(!dict_sys.sys_tables->not_redundant());
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_tables, sys_index,
 | |
| 			     DICT_FLD__SYS_TABLES__ID, "ID"));
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_tables, sys_index,
 | |
| 			     DICT_FLD__SYS_TABLES__N_COLS, "N_COLS"));
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_tables, sys_index,
 | |
| 			     DICT_FLD__SYS_TABLES__TYPE, "TYPE"));
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_tables, sys_index,
 | |
| 			     DICT_FLD__SYS_TABLES__MIX_LEN, "MIX_LEN"));
 | |
| 	ut_ad(name_of_col_is(dict_sys.sys_tables, sys_index,
 | |
| 			     DICT_FLD__SYS_TABLES__SPACE, "SPACE"));
 | |
| 
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 	dfield_set_data(&dfield, name.data(), name.size());
 | |
| 	dict_index_copy_types(&tuple, sys_index, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sys_index;
 | |
| 
 | |
| 	bool uncommitted = false;
 | |
| reload:
 | |
| 	mtr.start();
 | |
| 	dberr_t err = btr_pcur_open_on_user_rec(&tuple,
 | |
| 						BTR_SEARCH_LEAF, &pcur, &mtr);
 | |
| 
 | |
| 	if (err != DB_SUCCESS || !btr_pcur_is_on_user_rec(&pcur)) {
 | |
| 		/* Not found */
 | |
| err_exit:
 | |
| 		mtr.commit();
 | |
| 		DBUG_RETURN(nullptr);
 | |
| 	}
 | |
| 
 | |
| 	const rec_t* rec = btr_pcur_get_rec(&pcur);
 | |
| 
 | |
| 	/* Check if the table name in record is the searched one */
 | |
| 	if (rec_get_field_start_offs(rec, 1) != name.size()
 | |
|             || memcmp(name.data(), rec, name.size())) {
 | |
| 		goto err_exit;
 | |
| 	}
 | |
| 
 | |
| 	dict_table_t* table;
 | |
| 	if (const char* err_msg =
 | |
| 	    dict_load_table_low(&mtr, uncommitted, rec, &table)) {
 | |
| 		if (err_msg != dict_load_table_flags) {
 | |
| 			ib::error() << err_msg;
 | |
| 		}
 | |
| 		goto err_exit;
 | |
| 	}
 | |
| 	if (!table) {
 | |
| 		goto err_exit;
 | |
| 	}
 | |
| 
 | |
|         const unsigned use_uncommitted = uncommitted
 | |
| 		? 2
 | |
| 		: table->id == mach_read_from_8(
 | |
| 			rec + rec_get_field_start_offs(
 | |
| 				rec, DICT_FLD__SYS_TABLES__ID));
 | |
| 
 | |
| 	mtr.commit();
 | |
| 
 | |
| 	mem_heap_t* heap = mem_heap_create(32000);
 | |
| 
 | |
| 	dict_load_tablespace(table, ignore_err);
 | |
| 
 | |
| 	switch (dict_load_columns(table, use_uncommitted, heap)) {
 | |
| 	case DB_SUCCESS_LOCKED_REC:
 | |
| 		ut_ad(!uncommitted);
 | |
| 		uncommitted = true;
 | |
| 		dict_mem_table_free(table);
 | |
| 		mem_heap_free(heap);
 | |
| 		goto reload;
 | |
| 	case DB_SUCCESS:
 | |
| 		if (!dict_load_virtual(table, uncommitted)) {
 | |
| 			break;
 | |
| 		}
 | |
| 		/* fall through */
 | |
| 	default:
 | |
| 		dict_mem_table_free(table);
 | |
| 		mem_heap_free(heap);
 | |
| 		DBUG_RETURN(nullptr);
 | |
| 	}
 | |
| 
 | |
| 	dict_table_add_system_columns(table, heap);
 | |
| 
 | |
| 	table->can_be_evicted = true;
 | |
| 	table->add_to_cache();
 | |
| 
 | |
| 	mem_heap_empty(heap);
 | |
| 
 | |
| 	ut_ad(dict_tf2_is_valid(table->flags, table->flags2));
 | |
| 
 | |
| 	/* If there is no tablespace for the table then we only need to
 | |
| 	load the index definitions. So that we can IMPORT the tablespace
 | |
| 	later. When recovering table locks for resurrected incomplete
 | |
| 	transactions, the tablespace should exist, because DDL operations
 | |
| 	were not allowed while the table is being locked by a transaction. */
 | |
| 	dict_err_ignore_t index_load_err =
 | |
| 		!(ignore_err & DICT_ERR_IGNORE_RECOVER_LOCK)
 | |
| 		&& !table->is_readable()
 | |
| 		? DICT_ERR_IGNORE_ALL
 | |
| 		: ignore_err;
 | |
| 
 | |
| 	err = dict_load_indexes(table, uncommitted, heap, index_load_err);
 | |
| 
 | |
| 	if (err == DB_TABLE_CORRUPT) {
 | |
| 		/* Refuse to load the table if the table has a corrupted
 | |
| 		cluster index */
 | |
| 		ut_ad(index_load_err != DICT_ERR_IGNORE_DROP);
 | |
| 		ib::error() << "Refusing to load corrupted table "
 | |
| 			    << table->name;
 | |
| evict:
 | |
| 		dict_sys.remove(table);
 | |
| 		mem_heap_free(heap);
 | |
| 		DBUG_RETURN(nullptr);
 | |
| 	}
 | |
| 
 | |
| 	if (err != DB_SUCCESS || !table->is_readable()) {
 | |
| 	} else if (dict_index_t* pk = dict_table_get_first_index(table)) {
 | |
| 		ut_ad(pk->is_primary());
 | |
| 		if (pk->is_corrupted()
 | |
| 		    || pk->page >= table->space->get_size()) {
 | |
| corrupted:
 | |
| 			table->corrupted = true;
 | |
| 			table->file_unreadable = true;
 | |
| 			err = DB_TABLE_CORRUPT;
 | |
| 		} else if (table->space->id
 | |
| 			   && ignore_err == DICT_ERR_IGNORE_DROP) {
 | |
| 			/* Do not bother to load data from .ibd files
 | |
| 			only to delete the .ibd files. */
 | |
| 			goto corrupted;
 | |
| 		} else {
 | |
| 			mtr.start();
 | |
| 			bool ok = false;
 | |
| 			if (buf_block_t* b = buf_page_get(
 | |
| 				    page_id_t(table->space->id, pk->page),
 | |
| 				    table->space->zip_size(),
 | |
| 				    RW_S_LATCH, &mtr)) {
 | |
| 				switch (mach_read_from_2(FIL_PAGE_TYPE
 | |
| 							 + b->page.frame)) {
 | |
| 				case FIL_PAGE_INDEX:
 | |
| 				case FIL_PAGE_TYPE_INSTANT:
 | |
| 					ok = true;
 | |
| 				}
 | |
| 			}
 | |
| 			mtr.commit();
 | |
| 			if (!ok) {
 | |
| 				goto corrupted;
 | |
| 			}
 | |
| 
 | |
| 			if (table->supports_instant()) {
 | |
| 				err = btr_cur_instant_init(table);
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		ut_ad(ignore_err & DICT_ERR_IGNORE_INDEX);
 | |
| 		if (ignore_err != DICT_ERR_IGNORE_DROP) {
 | |
| 			err = DB_CORRUPTION;
 | |
| 			goto evict;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* Initialize table foreign_child value. Its value could be
 | |
| 	changed when dict_load_foreigns() is called below */
 | |
| 	table->fk_max_recusive_level = 0;
 | |
| 
 | |
| 	/* We will load the foreign key information only if
 | |
| 	all indexes were loaded. */
 | |
| 	if (!table->is_readable()) {
 | |
| 		/* Don't attempt to load the indexes from disk. */
 | |
| 	} else if (err == DB_SUCCESS) {
 | |
| 		auto i = fk_tables.size();
 | |
| 		err = dict_load_foreigns(table->name.m_name, nullptr,
 | |
| 					 0, true, ignore_err, fk_tables);
 | |
| 
 | |
| 		if (err != DB_SUCCESS) {
 | |
| 			fk_tables.erase(fk_tables.begin() + i, fk_tables.end());
 | |
| 			ib::warn() << "Load table " << table->name
 | |
| 				<< " failed, the table has missing"
 | |
| 				" foreign key indexes. Turn off"
 | |
| 				" 'foreign_key_checks' and try again.";
 | |
| 			goto evict;
 | |
| 		} else {
 | |
| 			dict_mem_table_fill_foreign_vcol_set(table);
 | |
| 			table->fk_max_recusive_level = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	mem_heap_free(heap);
 | |
| 
 | |
| 	ut_ad(!table
 | |
| 	      || (ignore_err & ~DICT_ERR_IGNORE_FK_NOKEY)
 | |
| 	      || !table->is_readable()
 | |
| 	      || !table->corrupted);
 | |
| 
 | |
| 	if (table && table->fts) {
 | |
| 		if (!(dict_table_has_fts_index(table)
 | |
| 		      || DICT_TF2_FLAG_IS_SET(table, DICT_TF2_FTS_HAS_DOC_ID)
 | |
| 		      || DICT_TF2_FLAG_IS_SET(table, DICT_TF2_FTS_ADD_DOC_ID))) {
 | |
| 			/* the table->fts could be created in dict_load_column
 | |
| 			when a user defined FTS_DOC_ID is present, but no
 | |
| 			FTS */
 | |
| 			table->fts->~fts_t();
 | |
| 			table->fts = nullptr;
 | |
| 		} else if (fts_optimize_wq) {
 | |
| 			fts_optimize_add_table(table);
 | |
| 		} else if (table->can_be_evicted) {
 | |
| 			/* fts_optimize_thread is not started yet.
 | |
| 			So make the table as non-evictable from cache. */
 | |
| 			dict_sys.prevent_eviction(table);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ut_ad(err != DB_SUCCESS || dict_foreign_set_validate(*table));
 | |
| 
 | |
| 	DBUG_RETURN(table);
 | |
| }
 | |
| 
 | |
| dict_table_t *dict_sys_t::load_table(const span<const char> &name,
 | |
|                                      dict_err_ignore_t ignore) noexcept
 | |
| {
 | |
|   if (dict_table_t *table= find_table(name))
 | |
|     return table;
 | |
|   dict_names_t fk_list;
 | |
|   dict_table_t *table= dict_load_table_one(name, ignore, fk_list);
 | |
|   while (!fk_list.empty())
 | |
|   {
 | |
|     const char *f= fk_list.front();
 | |
|     const span<const char> name{f, strlen(f)};
 | |
|     if (!find_table(name))
 | |
|       dict_load_table_one(name, ignore, fk_list);
 | |
|     fk_list.pop_front();
 | |
|   }
 | |
| 
 | |
|   return table;
 | |
| }
 | |
| 
 | |
| /***********************************************************************//**
 | |
| Loads a table object based on the table id.
 | |
| @return table; NULL if table does not exist */
 | |
| dict_table_t*
 | |
| dict_load_table_on_id(
 | |
| /*==================*/
 | |
| 	table_id_t		table_id,	/*!< in: table id */
 | |
| 	dict_err_ignore_t	ignore_err)	/*!< in: errors to ignore
 | |
| 						when loading the table */
 | |
| {
 | |
| 	byte		id_buf[8];
 | |
| 	btr_pcur_t	pcur;
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 	mtr_t		mtr;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	/* NOTE that the operation of this function is protected by
 | |
| 	dict_sys.latch, and therefore no deadlocks can occur
 | |
| 	with other dictionary operations. */
 | |
| 
 | |
| 	mtr.start();
 | |
| 	/*---------------------------------------------------*/
 | |
| 	/* Get the secondary index based on ID for table SYS_TABLES */
 | |
| 	dict_index_t *sys_table_ids =
 | |
| 		dict_sys.sys_tables->indexes.start->indexes.next;
 | |
| 
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 
 | |
| 	/* Write the table id in byte format to id_buf */
 | |
| 	mach_write_to_8(id_buf, table_id);
 | |
| 	dfield_set_data(&dfield, id_buf, 8);
 | |
| 	dict_index_copy_types(&tuple, sys_table_ids, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sys_table_ids;
 | |
| 
 | |
| 	dict_table_t* table = nullptr;
 | |
| 
 | |
| 	if (btr_pcur_open_on_user_rec(&tuple, BTR_SEARCH_LEAF, &pcur, &mtr)
 | |
| 	    == DB_SUCCESS
 | |
| 	    && btr_pcur_is_on_user_rec(&pcur)) {
 | |
| 		/*---------------------------------------------------*/
 | |
| 		/* Now we have the record in the secondary index
 | |
| 		containing the table ID and NAME */
 | |
| 		const rec_t* rec = btr_pcur_get_rec(&pcur);
 | |
| check_rec:
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_TABLE_IDS__ID, &len);
 | |
| 		ut_ad(len == 8);
 | |
| 
 | |
| 		/* Check if the table id in record is the one searched for */
 | |
| 		if (table_id == mach_read_from_8(field)) {
 | |
| 			field = rec_get_nth_field_old(rec,
 | |
| 				DICT_FLD__SYS_TABLE_IDS__NAME, &len);
 | |
| 			table = dict_sys.load_table(
 | |
| 				{reinterpret_cast<const char*>(field),
 | |
| 				 len}, ignore_err);
 | |
| 			if (table && table->id != table_id) {
 | |
| 				ut_ad(rec_get_deleted_flag(rec, 0));
 | |
| 				table = nullptr;
 | |
| 			}
 | |
| 			if (!table) {
 | |
| 				while (btr_pcur_move_to_next(&pcur, &mtr)) {
 | |
| 					rec = btr_pcur_get_rec(&pcur);
 | |
| 
 | |
| 					if (page_rec_is_user_rec(rec)) {
 | |
| 						goto check_rec;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	mtr.commit();
 | |
| 	return table;
 | |
| }
 | |
| 
 | |
| /********************************************************************//**
 | |
| This function is called when the database is booted. Loads system table
 | |
| index definitions except for the clustered index which is added to the
 | |
| dictionary cache at booting before calling this function. */
 | |
| void
 | |
| dict_load_sys_table(
 | |
| /*================*/
 | |
| 	dict_table_t*	table)	/*!< in: system table */
 | |
| {
 | |
| 	mem_heap_t*	heap;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	heap = mem_heap_create(1000);
 | |
| 
 | |
| 	dict_load_indexes(table, false, heap, DICT_ERR_IGNORE_NONE);
 | |
| 
 | |
| 	mem_heap_free(heap);
 | |
| }
 | |
| 
 | |
| MY_ATTRIBUTE((nonnull, warn_unused_result))
 | |
| /********************************************************************//**
 | |
| Loads foreign key constraint col names (also for the referenced table).
 | |
| Members that must be set (and valid) in foreign:
 | |
| foreign->heap
 | |
| foreign->n_fields
 | |
| foreign->id ('\0'-terminated)
 | |
| Members that will be created and set by this function:
 | |
| foreign->foreign_col_names[i]
 | |
| foreign->referenced_col_names[i]
 | |
| (for i=0..foreign->n_fields-1) */
 | |
| static dberr_t dict_load_foreign_cols(dict_foreign_t *foreign, trx_id_t trx_id)
 | |
| {
 | |
| 	btr_pcur_t	pcur;
 | |
| 	mtr_t		mtr;
 | |
| 	size_t		id_len;
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	id_len = strlen(foreign->id);
 | |
| 
 | |
| 	foreign->foreign_col_names = static_cast<const char**>(
 | |
| 		mem_heap_alloc(foreign->heap,
 | |
| 			       foreign->n_fields * sizeof(void*)));
 | |
| 
 | |
| 	foreign->referenced_col_names = static_cast<const char**>(
 | |
| 		mem_heap_alloc(foreign->heap,
 | |
| 			       foreign->n_fields * sizeof(void*)));
 | |
| 
 | |
| 	mtr.start();
 | |
| 
 | |
| 	dict_index_t* sys_index = dict_sys.sys_foreign_cols->indexes.start;
 | |
| 	ut_ad(!dict_sys.sys_foreign_cols->not_redundant());
 | |
| 
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 
 | |
| 	dfield_set_data(&dfield, foreign->id, id_len);
 | |
| 	dict_index_copy_types(&tuple, sys_index, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sys_index;
 | |
| 
 | |
| 	mem_heap_t* heap = nullptr;
 | |
| 	dberr_t err = btr_pcur_open_on_user_rec(&tuple,
 | |
| 						BTR_SEARCH_LEAF, &pcur, &mtr);
 | |
| 	if (err != DB_SUCCESS) {
 | |
| 		goto func_exit;
 | |
| 	}
 | |
| 	for (ulint i = 0; i < foreign->n_fields; i++) {
 | |
| 		ut_a(btr_pcur_is_on_user_rec(&pcur));
 | |
| 
 | |
| 		const rec_t* rec = btr_pcur_get_rec(&pcur);
 | |
| 		ulint len;
 | |
| 		const byte* field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_FOREIGN_COLS__DB_TRX_ID, &len);
 | |
| 		ut_a(len == DATA_TRX_ID_LEN);
 | |
| 
 | |
| 		if (UNIV_LIKELY_NULL(heap)) {
 | |
| 			mem_heap_empty(heap);
 | |
| 		}
 | |
| 
 | |
| 		const trx_id_t id = trx_read_trx_id(field);
 | |
| 		if (!id) {
 | |
| 		} else if (id != trx_id && trx_sys.find(nullptr, id, false)) {
 | |
| 			const auto savepoint = mtr.get_savepoint();
 | |
| 			rec_offs* offsets = rec_get_offsets(
 | |
| 				rec, sys_index, nullptr, true, ULINT_UNDEFINED,
 | |
| 				&heap);
 | |
| 			const rec_t* old_vers;
 | |
| 			row_vers_build_for_semi_consistent_read(
 | |
| 				nullptr, rec, &mtr, sys_index, &offsets, &heap,
 | |
| 				heap, &old_vers, nullptr);
 | |
| 			mtr.rollback_to_savepoint(savepoint);
 | |
| 			rec = old_vers;
 | |
| 			if (!rec || rec_get_deleted_flag(rec, 0)) {
 | |
| 				goto next;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (rec_get_deleted_flag(rec, 0)) {
 | |
| 			ut_ad(id);
 | |
| 			goto next;
 | |
| 		}
 | |
| 
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_FOREIGN_COLS__ID, &len);
 | |
| 
 | |
| 		if (len != id_len || memcmp(foreign->id, field, len)) {
 | |
| 			const rec_t*	pos;
 | |
| 			ulint		pos_len;
 | |
| 			const rec_t*	for_col_name;
 | |
| 			ulint		for_col_name_len;
 | |
| 			const rec_t*	ref_col_name;
 | |
| 			ulint		ref_col_name_len;
 | |
| 
 | |
| 			pos = rec_get_nth_field_old(
 | |
| 				rec, DICT_FLD__SYS_FOREIGN_COLS__POS,
 | |
| 				&pos_len);
 | |
| 
 | |
| 			for_col_name = rec_get_nth_field_old(
 | |
| 				rec, DICT_FLD__SYS_FOREIGN_COLS__FOR_COL_NAME,
 | |
| 				&for_col_name_len);
 | |
| 
 | |
| 			ref_col_name = rec_get_nth_field_old(
 | |
| 				rec, DICT_FLD__SYS_FOREIGN_COLS__REF_COL_NAME,
 | |
| 				&ref_col_name_len);
 | |
| 
 | |
| 			ib::error	sout;
 | |
| 
 | |
| 			sout << "Unable to load column names for foreign"
 | |
| 				" key '" << foreign->id
 | |
| 				<< "' because it was not found in"
 | |
| 				" InnoDB internal table SYS_FOREIGN_COLS. The"
 | |
| 				" closest entry we found is:"
 | |
| 				" (ID='";
 | |
| 			sout.write(field, len);
 | |
| 			sout << "', POS=" << mach_read_from_4(pos)
 | |
| 				<< ", FOR_COL_NAME='";
 | |
| 			sout.write(for_col_name, for_col_name_len);
 | |
| 			sout << "', REF_COL_NAME='";
 | |
| 			sout.write(ref_col_name, ref_col_name_len);
 | |
| 			sout << "')";
 | |
| 
 | |
| 			err = DB_CORRUPTION;
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_FOREIGN_COLS__POS, &len);
 | |
| 		ut_a(len == 4);
 | |
| 		ut_a(i == mach_read_from_4(field));
 | |
| 
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_FOREIGN_COLS__FOR_COL_NAME, &len);
 | |
| 		foreign->foreign_col_names[i] = mem_heap_strdupl(
 | |
| 			foreign->heap, (char*) field, len);
 | |
| 
 | |
| 		field = rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_FOREIGN_COLS__REF_COL_NAME, &len);
 | |
| 		foreign->referenced_col_names[i] = mem_heap_strdupl(
 | |
| 			foreign->heap, (char*) field, len);
 | |
| 
 | |
| next:
 | |
| 		btr_pcur_move_to_next_user_rec(&pcur, &mtr);
 | |
| 	}
 | |
| func_exit:
 | |
| 	mtr.commit();
 | |
| 	if (UNIV_LIKELY_NULL(heap)) {
 | |
| 		mem_heap_free(heap);
 | |
| 	}
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| /***********************************************************************//**
 | |
| Loads a foreign key constraint to the dictionary cache. If the referenced
 | |
| table is not yet loaded, it is added in the output parameter (fk_tables).
 | |
| @return DB_SUCCESS or error code */
 | |
| static MY_ATTRIBUTE((warn_unused_result))
 | |
| dberr_t
 | |
| dict_load_foreign(
 | |
| /*==============*/
 | |
| 	const char*		table_name,	/*!< in: table name */
 | |
| 	bool			uncommitted,	/*!< in: use READ UNCOMMITTED
 | |
| 						transaction isolation level */
 | |
| 	const char**		col_names,
 | |
| 				/*!< in: column names, or NULL
 | |
| 				to use foreign->foreign_table->col_names */
 | |
| 	trx_id_t		trx_id,
 | |
| 				/*!< in: current transaction id, or 0 */
 | |
| 	bool			check_recursive,
 | |
| 				/*!< in: whether to record the foreign table
 | |
| 				parent count to avoid unlimited recursive
 | |
| 				load of chained foreign tables */
 | |
| 	bool			check_charsets,
 | |
| 				/*!< in: whether to check charset
 | |
| 				compatibility */
 | |
| 	span<const char>	id,
 | |
| 				/*!< in: foreign constraint id */
 | |
| 	dict_err_ignore_t	ignore_err,
 | |
| 				/*!< in: error to be ignored */
 | |
| 	dict_names_t&	fk_tables)
 | |
| 				/*!< out: the foreign key constraint is added
 | |
| 				to the dictionary cache only if the referenced
 | |
| 				table is already in cache.  Otherwise, the
 | |
| 				foreign key constraint is not added to cache,
 | |
| 				and the referenced table is added to this
 | |
| 				stack. */
 | |
| {
 | |
| 	dict_foreign_t*	foreign;
 | |
| 	btr_pcur_t	pcur;
 | |
| 	const byte*	field;
 | |
| 	ulint		len;
 | |
| 	mtr_t		mtr;
 | |
| 	dict_table_t*	for_table;
 | |
| 	dict_table_t*	ref_table;
 | |
| 
 | |
| 	DBUG_ENTER("dict_load_foreign");
 | |
| 	DBUG_PRINT("dict_load_foreign",
 | |
| 		   ("id: '%.*s', check_recursive: %d",
 | |
| 		    int(id.size()), id.data(), check_recursive));
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	dict_index_t* sys_index = dict_sys.sys_foreign->indexes.start;
 | |
| 	ut_ad(!dict_sys.sys_foreign->not_redundant());
 | |
| 
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 	dfield_set_data(&dfield, id.data(), id.size());
 | |
| 	dict_index_copy_types(&tuple, sys_index, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sys_index;
 | |
| 
 | |
| 	mtr.start();
 | |
| 
 | |
| 	mem_heap_t* heap = nullptr;
 | |
| 	dberr_t err = btr_pcur_open_on_user_rec(&tuple,
 | |
| 						BTR_SEARCH_LEAF, &pcur, &mtr);
 | |
| 	if (err != DB_SUCCESS) {
 | |
| 		goto err_exit;
 | |
| 	}
 | |
| 
 | |
| 	if (!btr_pcur_is_on_user_rec(&pcur)) {
 | |
| not_found:
 | |
| 		err = DB_NOT_FOUND;
 | |
| err_exit:
 | |
| 		mtr.commit();
 | |
| 		if (UNIV_LIKELY_NULL(heap)) {
 | |
| 			mem_heap_free(heap);
 | |
| 		}
 | |
| 		DBUG_RETURN(err);
 | |
| 	}
 | |
| 
 | |
| 	const rec_t* rec = btr_pcur_get_rec(&pcur);
 | |
| 	static_assert(DICT_FLD__SYS_FOREIGN__ID == 0, "compatibility");
 | |
| 	field = rec_get_nth_field_old(rec, DICT_FLD__SYS_FOREIGN__ID, &len);
 | |
| 
 | |
| 	/* Check if the id in record is the searched one */
 | |
| 	if (len != id.size() || memcmp(id.data(), field, id.size())) {
 | |
| 		goto not_found;
 | |
| 	}
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__DB_TRX_ID, &len);
 | |
| 	ut_a(len == DATA_TRX_ID_LEN);
 | |
| 
 | |
| 	const trx_id_t tid = trx_read_trx_id(field);
 | |
| 
 | |
| 	if (tid && tid != trx_id && !uncommitted
 | |
| 	    && trx_sys.find(nullptr, tid, false)) {
 | |
| 		const auto savepoint = mtr.get_savepoint();
 | |
| 		rec_offs* offsets = rec_get_offsets(
 | |
| 			rec, sys_index, nullptr, true, ULINT_UNDEFINED, &heap);
 | |
| 		const rec_t* old_vers;
 | |
| 		row_vers_build_for_semi_consistent_read(
 | |
| 			nullptr, rec, &mtr, sys_index, &offsets, &heap,
 | |
| 			heap, &old_vers, nullptr);
 | |
| 		mtr.rollback_to_savepoint(savepoint);
 | |
| 		rec = old_vers;
 | |
| 		if (!rec) {
 | |
| 			goto not_found;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (rec_get_deleted_flag(rec, 0)) {
 | |
| 		ut_ad(tid);
 | |
| 		goto not_found;
 | |
| 	}
 | |
| 
 | |
| 	/* Read the table names and the number of columns associated
 | |
| 	with the constraint */
 | |
| 
 | |
| 	foreign = dict_mem_foreign_create();
 | |
| 
 | |
| 	uint32_t n_fields_and_type = mach_read_from_4(
 | |
| 		rec_get_nth_field_old(
 | |
| 			rec, DICT_FLD__SYS_FOREIGN__N_COLS, &len));
 | |
| 
 | |
| 	ut_a(len == 4);
 | |
| 
 | |
| 	/* We store the type in the bits 24..29 of n_fields_and_type. */
 | |
| 
 | |
| 	foreign->type = (n_fields_and_type >> 24) & ((1U << 6) - 1);
 | |
| 	foreign->n_fields = n_fields_and_type & dict_index_t::MAX_N_FIELDS;
 | |
| 
 | |
| 	foreign->id = mem_heap_strdupl(foreign->heap, id.data(), id.size());
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__FOR_NAME, &len);
 | |
| 
 | |
| 	foreign->foreign_table_name = mem_heap_strdupl(
 | |
| 		foreign->heap, (char*) field, len);
 | |
| 	dict_mem_foreign_table_name_lookup_set(foreign, TRUE);
 | |
| 
 | |
| 	const size_t foreign_table_name_len = len;
 | |
| 	const size_t table_name_len = strlen(table_name);
 | |
| 
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN__REF_NAME, &len);
 | |
| 
 | |
| 	if (!my_charset_latin1.strnncoll(table_name, table_name_len,
 | |
| 					 foreign->foreign_table_name,
 | |
| 					 foreign_table_name_len)) {
 | |
| 	} else if (!check_recursive
 | |
| 		   && !my_charset_latin1.strnncoll(table_name, table_name_len,
 | |
| 						   (const char*) field, len)) {
 | |
| 	} else {
 | |
| 		dict_foreign_free(foreign);
 | |
| 		goto not_found;
 | |
| 	}
 | |
| 
 | |
| 	foreign->referenced_table_name = mem_heap_strdupl(
 | |
| 		foreign->heap, (const char*) field, len);
 | |
| 	dict_mem_referenced_table_name_lookup_set(foreign, TRUE);
 | |
| 
 | |
| 	mtr.commit();
 | |
| 	if (UNIV_LIKELY_NULL(heap)) {
 | |
| 		mem_heap_free(heap);
 | |
| 	}
 | |
| 
 | |
| 	err = dict_load_foreign_cols(foreign, trx_id);
 | |
| 	if (err != DB_SUCCESS) {
 | |
| 		goto load_error;
 | |
| 	}
 | |
| 
 | |
| 	ref_table = dict_sys.find_table(
 | |
| 		{foreign->referenced_table_name_lookup,
 | |
| 		 strlen(foreign->referenced_table_name_lookup)});
 | |
| 	for_table = dict_sys.find_table(
 | |
| 		{foreign->foreign_table_name_lookup,
 | |
| 		 strlen(foreign->foreign_table_name_lookup)});
 | |
| 
 | |
| 	if (!for_table) {
 | |
| 		/* To avoid recursively loading the tables related through
 | |
| 		the foreign key constraints, the child table name is saved
 | |
| 		here.  The child table will be loaded later, along with its
 | |
| 		foreign key constraint. */
 | |
| 
 | |
| 		ut_a(ref_table != NULL);
 | |
| 		fk_tables.push_back(
 | |
| 			mem_heap_strdupl(ref_table->heap,
 | |
| 					 foreign->foreign_table_name_lookup,
 | |
| 					 foreign_table_name_len));
 | |
| load_error:
 | |
| 		dict_foreign_remove_from_cache(foreign);
 | |
| 		DBUG_RETURN(err);
 | |
| 	}
 | |
| 
 | |
| 	ut_a(for_table || ref_table);
 | |
| 
 | |
| 	/* Note that there may already be a foreign constraint object in
 | |
| 	the dictionary cache for this constraint: then the following
 | |
| 	call only sets the pointers in it to point to the appropriate table
 | |
| 	and index objects and frees the newly created object foreign.
 | |
| 	Adding to the cache should always succeed since we are not creating
 | |
| 	a new foreign key constraint but loading one from the data
 | |
| 	dictionary. */
 | |
| 
 | |
| 	DBUG_RETURN(dict_foreign_add_to_cache(foreign, col_names,
 | |
| 					      check_charsets,
 | |
| 					      ignore_err));
 | |
| }
 | |
| 
 | |
| /***********************************************************************//**
 | |
| Loads foreign key constraints where the table is either the foreign key
 | |
| holder or where the table is referenced by a foreign key. Adds these
 | |
| constraints to the data dictionary.
 | |
| 
 | |
| The foreign key constraint is loaded only if the referenced table is also
 | |
| in the dictionary cache.  If the referenced table is not in dictionary
 | |
| cache, then it is added to the output parameter (fk_tables).
 | |
| 
 | |
| @return DB_SUCCESS or error code */
 | |
| dberr_t
 | |
| dict_load_foreigns(
 | |
| 	const char*		table_name,	/*!< in: table name */
 | |
| 	const char**		col_names,	/*!< in: column names, or NULL
 | |
| 						to use table->col_names */
 | |
| 	trx_id_t		trx_id,		/*!< in: DDL transaction id,
 | |
| 						or 0 to check
 | |
| 						recursive load of tables
 | |
| 						chained by FK */
 | |
| 	bool			check_charsets,	/*!< in: whether to check
 | |
| 						charset compatibility */
 | |
| 	dict_err_ignore_t	ignore_err,	/*!< in: error to be ignored */
 | |
| 	dict_names_t&		fk_tables)
 | |
| 						/*!< out: stack of table
 | |
| 						names which must be loaded
 | |
| 						subsequently to load all the
 | |
| 						foreign key constraints. */
 | |
| {
 | |
| 	btr_pcur_t	pcur;
 | |
| 	mtr_t		mtr;
 | |
| 
 | |
| 	DBUG_ENTER("dict_load_foreigns");
 | |
| 
 | |
| 	ut_ad(dict_sys.locked());
 | |
| 
 | |
| 	if (!dict_sys.sys_foreign || !dict_sys.sys_foreign_cols) {
 | |
| 		if (ignore_err & DICT_ERR_IGNORE_FK_NOKEY) {
 | |
| 			DBUG_RETURN(DB_SUCCESS);
 | |
| 		}
 | |
| 		sql_print_information("InnoDB: No foreign key system tables"
 | |
| 				      " in the database");
 | |
| 		DBUG_RETURN(DB_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	ut_ad(!dict_sys.sys_foreign->not_redundant());
 | |
| 
 | |
| 	dict_index_t *sec_index = dict_table_get_next_index(
 | |
| 		dict_table_get_first_index(dict_sys.sys_foreign));
 | |
| 	ut_ad(!strcmp(sec_index->fields[0].name, "FOR_NAME"));
 | |
| 	bool check_recursive = !trx_id;
 | |
| 	dfield_t dfield;
 | |
| 	dtuple_t tuple{
 | |
| 		0,1,1,&dfield,0,nullptr
 | |
| #ifdef UNIV_DEBUG
 | |
| 		, DATA_TUPLE_MAGIC_N
 | |
| #endif
 | |
| 	};
 | |
| 
 | |
| start_load:
 | |
| 	mtr.start();
 | |
| 	dfield_set_data(&dfield, table_name, strlen(table_name));
 | |
| 	dict_index_copy_types(&tuple, sec_index, 1);
 | |
| 	pcur.btr_cur.page_cur.index = sec_index;
 | |
| 
 | |
| 	dberr_t err = btr_pcur_open_on_user_rec(&tuple,
 | |
| 						BTR_SEARCH_LEAF, &pcur, &mtr);
 | |
| 	if (err != DB_SUCCESS) {
 | |
| 		DBUG_RETURN(err);
 | |
| 	}
 | |
| loop:
 | |
| 	const rec_t* rec = btr_pcur_get_rec(&pcur);
 | |
| 	const byte* field;
 | |
| 	const auto maybe_deleted = rec_get_deleted_flag(rec, 0);
 | |
| 
 | |
| 	if (!btr_pcur_is_on_user_rec(&pcur)) {
 | |
| 		/* End of index */
 | |
| 
 | |
| 		goto load_next_index;
 | |
| 	}
 | |
| 
 | |
| 	/* Now we have the record in the secondary index containing a table
 | |
| 	name and a foreign constraint ID */
 | |
| 
 | |
| 	ulint len;
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_FOR_NAME__NAME, &len);
 | |
| 
 | |
| 	/* Check if the table name in the record is the one searched for; the
 | |
| 	following call does the comparison in the latin1_swedish_ci
 | |
| 	charset-collation, in a case-insensitive way. */
 | |
| 
 | |
| 	if (cmp_data(dfield_get_type(&dfield)->mtype,
 | |
| 		     dfield_get_type(&dfield)->prtype,
 | |
| 		     false,
 | |
| 		     reinterpret_cast<const byte*>(table_name),
 | |
| 		     dfield_get_len(&dfield),
 | |
| 		     field, len)) {
 | |
| 		goto load_next_index;
 | |
| 	}
 | |
| 
 | |
| 	/* Since table names in SYS_FOREIGN are stored in a case-insensitive
 | |
| 	order, we have to check that the table name matches also in a binary
 | |
| 	string comparison. On Unix, MySQL allows table names that only differ
 | |
| 	in character case.  If lower_case_table_names=2 then what is stored
 | |
| 	may not be the same case, but the previous comparison showed that they
 | |
| 	match with no-case.  */
 | |
| 
 | |
| 	if (lower_case_table_names != 2 && memcmp(field, table_name, len)) {
 | |
| 		goto next_rec;
 | |
| 	}
 | |
| 
 | |
| 	/* Now we get a foreign key constraint id */
 | |
| 	field = rec_get_nth_field_old(
 | |
| 		rec, DICT_FLD__SYS_FOREIGN_FOR_NAME__ID, &len);
 | |
| 
 | |
| 	/* Copy the string because the page may be modified or evicted
 | |
| 	after mtr.commit() below. */
 | |
| 	char	fk_id[MAX_TABLE_NAME_LEN + NAME_LEN];
 | |
| 	err = DB_SUCCESS;
 | |
| 	if (UNIV_LIKELY(len < sizeof fk_id)) {
 | |
| 		memcpy(fk_id, field, len);
 | |
| 	}
 | |
| 
 | |
| 	btr_pcur_store_position(&pcur, &mtr);
 | |
| 
 | |
| 	mtr.commit();
 | |
| 
 | |
| 	/* Load the foreign constraint definition to the dictionary cache */
 | |
| 
 | |
| 	err = len < sizeof fk_id
 | |
| 		? dict_load_foreign(table_name, false, col_names, trx_id,
 | |
| 				    check_recursive, check_charsets,
 | |
| 				    {fk_id, len}, ignore_err, fk_tables)
 | |
| 		: DB_CORRUPTION;
 | |
| 
 | |
| 	switch (err) {
 | |
| 	case DB_SUCCESS:
 | |
| 		break;
 | |
| 	case DB_NOT_FOUND:
 | |
| 		if (maybe_deleted) {
 | |
| 			break;
 | |
| 		}
 | |
| 		sql_print_error("InnoDB: Cannot load foreign constraint %.*s:"
 | |
| 				" could not find the relevant record in "
 | |
| 				"SYS_FOREIGN", int(len), fk_id);
 | |
| 		/* fall through */
 | |
| 	default:
 | |
| corrupted:
 | |
| 		ut_free(pcur.old_rec_buf);
 | |
| 		DBUG_RETURN(err);
 | |
| 	}
 | |
| 
 | |
| 	mtr.start();
 | |
| 	if (pcur.restore_position(BTR_SEARCH_LEAF, &mtr)
 | |
| 	    == btr_pcur_t::CORRUPTED) {
 | |
| 		mtr.commit();
 | |
| 		goto corrupted;
 | |
| 	}
 | |
| next_rec:
 | |
| 	btr_pcur_move_to_next_user_rec(&pcur, &mtr);
 | |
| 
 | |
| 	goto loop;
 | |
| 
 | |
| load_next_index:
 | |
| 	mtr.commit();
 | |
| 
 | |
| 	if ((sec_index = dict_table_get_next_index(sec_index))) {
 | |
| 		/* Switch to scan index on REF_NAME, fk_max_recusive_level
 | |
| 		already been updated when scanning FOR_NAME index, no need to
 | |
| 		update again */
 | |
| 		check_recursive = false;
 | |
| 		goto start_load;
 | |
| 	}
 | |
| 
 | |
| 	ut_free(pcur.old_rec_buf);
 | |
| 	DBUG_RETURN(DB_SUCCESS);
 | |
| }
 | 
