polardbxengine/sql/dd/impl/bootstrap/bootstrapper.cc

1830 lines
72 KiB
C++

/* Copyright (c) 2014, 2019, Oracle and/or its affiliates. All rights reserved.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License, version 2.0,
as published by the Free Software Foundation.
This program is also distributed with certain software (including
but not limited to OpenSSL) that is licensed under separate terms,
as designated in a particular file or component or in included license
documentation. The authors of MySQL hereby grant you an additional
permission to link the program and your derivative works with the
separately licensed software that they have included with MySQL.
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, version 2.0, 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 St, Fifth Floor, Boston, MA 02110-1301 USA */
#include "sql/dd/impl/bootstrap/bootstrapper.h"
#include <stddef.h>
#include <sys/types.h>
#include <memory>
#include <new>
#include <set>
#include <string>
#include <utility>
#include <vector>
#include "m_ctype.h"
#include "my_dbug.h"
#include "my_loglevel.h"
#include "my_sys.h"
#include "mysql/components/services/log_builtins.h"
#include "mysql_version.h" // MYSQL_VERSION_ID
#include "mysqld_error.h"
#include "sql/auth/sql_security_ctx.h"
#include "sql/dd/cache/dictionary_client.h" // dd::cache::Dictionary_client
#include "sql/dd/dd.h" // dd::create_object
#include "sql/dd/impl/bootstrap/bootstrap_ctx.h" // DD_bootstrap_ctx
#include "sql/dd/impl/cache/shared_dictionary_cache.h" // Shared_dictionary_cache
#include "sql/dd/impl/cache/storage_adapter.h" // Storage_adapter
#include "sql/dd/impl/dictionary_impl.h" // dd::Dictionary_impl
#include "sql/dd/impl/raw/object_keys.h"
#include "sql/dd/impl/sdi.h" // dd::sdi::store
#include "sql/dd/impl/tables/character_sets.h" // dd::tables::Character_sets
#include "sql/dd/impl/tables/collations.h" // dd::tables::Collations
#include "sql/dd/impl/tables/dd_properties.h" // dd::tables::DD_properties
#include "sql/dd/impl/tables/tables.h" // dd::tables::Tables
#include "sql/dd/impl/types/schema_impl.h" // dd::Schema_impl
#include "sql/dd/impl/types/table_impl.h" // dd::Table_impl
#include "sql/dd/impl/types/tablespace_impl.h" // dd::Table_impl
#include "sql/dd/impl/upgrade/dd.h" // dd::upgrade::upgrade_tables
#include "sql/dd/impl/upgrade/server.h" // dd::upgrade::do_server_upgrade_checks
#include "sql/dd/impl/utils.h" // dd::execute_query
#include "sql/dd/object_id.h"
#include "sql/dd/types/abstract_table.h"
#include "sql/dd/types/object_table.h" // dd::Object_table
#include "sql/dd/types/object_table_definition.h" // dd::Object_table_definition
#include "sql/dd/types/table.h"
#include "sql/dd/types/tablespace.h"
#include "sql/dd/types/tablespace_file.h" // dd::Tablespace_file
#include "sql/dd/upgrade/server.h" // UPGRADE_NONE
#include "sql/dd/upgrade_57/upgrade.h" // upgrade_57::Upgrade_status
#include "sql/handler.h" // dict_init_mode_t
#include "sql/mdl.h"
#include "sql/mysqld.h"
#include "sql/sd_notify.h" // sysd::notify
#include "sql/thd_raii.h"
using namespace dd;
///////////////////////////////////////////////////////////////////////////
namespace {
// Initialize recovery in the DDSE.
bool DDSE_dict_recover(THD *thd, dict_recovery_mode_t dict_recovery_mode,
uint version) {
handlerton *ddse = ha_resolve_by_legacy_type(thd, DB_TYPE_INNODB);
if (ddse->dict_recover == nullptr) return true;
bool error = ddse->dict_recover(dict_recovery_mode, version);
/*
Commit when tablespaces have been initialized, since in that
case, tablespace meta data is added.
*/
if (dict_recovery_mode == DICT_RECOVERY_INITIALIZE_TABLESPACES)
return dd::end_transaction(thd, error);
return error;
}
/*
Update the System_tables registry with meta data from 'dd_properties'.
Iterate over the tables in the DD_properties. If this is minor downgrade,
add new tables that were added in the newer version to the System_tables
registry. If this is not minor downgrade, assert that all tables in the
DD_properties indeed have a corresponding entry in the System_tables
registry.
*/
bool update_system_tables(THD *thd) {
std::unique_ptr<dd::Properties> system_tables_props;
bool exists = false;
if (dd::tables::DD_properties::instance().get(
thd, "SYSTEM_TABLES", &system_tables_props, &exists) ||
!exists) {
my_error(ER_DD_INIT_FAILED, MYF(0));
return true;
}
/*
We would normally use a range based loop here, but the developerstudio
compiler on Solaris does not handle this when the base collection has
pure virtual begin() and end() functions.
*/
for (Properties::const_iterator it = system_tables_props->begin();
it != system_tables_props->end(); ++it) {
// Check if this is a CORE, INERT, SECOND or DDSE table.
if (!dd::get_dictionary()->is_dd_table_name(MYSQL_SCHEMA_NAME.str,
it->first)) {
if (bootstrap::DD_bootstrap_ctx::instance().is_minor_downgrade()) {
/*
Add tables as type CORE regardless of the actual type, which
is irrelevant in this case.
*/
System_tables::instance()->add(MYSQL_SCHEMA_NAME.str, it->first,
System_tables::Types::CORE, nullptr);
} else {
my_error(ER_DD_METADATA_NOT_FOUND, MYF(0), it->first.c_str());
return true;
}
} else {
/*
The table is a known DD table. Then, we get its definition
and add it to the Object_table instance. The definition might
not exist if the table was added after the version that we
are upgrading from.
*/
String_type tbl_prop_str;
if (!system_tables_props->exists(it->first) ||
system_tables_props->get(it->first, &tbl_prop_str))
continue;
const Object_table *table_def = System_tables::instance()->find_table(
MYSQL_SCHEMA_NAME.str, it->first);
DBUG_ASSERT(table_def);
std::unique_ptr<dd::Properties> tbl_props(
Properties::parse_properties(tbl_prop_str));
String_type def;
if (tbl_props->get(dd::tables::DD_properties::dd_key(
dd::tables::DD_properties::DD_property::DEF),
&def)) {
my_error(ER_DD_METADATA_NOT_FOUND, MYF(0), it->first.c_str());
return true;
}
std::unique_ptr<Properties> table_def_properties(
Properties::parse_properties(def));
table_def->set_actual_table_definition(*table_def_properties);
}
}
return false;
}
// Create a DD table using the target table definition.
bool create_target_table(THD *thd, const Object_table *object_table) {
DBUG_ASSERT(object_table != nullptr);
/*
The target table definition may not be present if the table
is abandoned. That's ok, not an error.
*/
if (object_table->is_abandoned()) return false;
String_type target_ddl_statement("");
const Object_table_definition *target_table_def =
object_table->target_table_definition();
DBUG_ASSERT(target_table_def != nullptr);
target_ddl_statement = target_table_def->get_ddl();
DBUG_ASSERT(!target_ddl_statement.empty());
return dd::execute_query(thd, target_ddl_statement);
}
// Create a DD table using the actual table definition.
/* purecov: begin inspected */
bool create_actual_table(THD *thd, const Object_table *object_table) {
/*
For minor downgrade, tables might have been added in the upgraded
server that we do not have any Object_table instance for. In that
case, we just skip them.
*/
if (object_table == nullptr) {
DBUG_ASSERT(bootstrap::DD_bootstrap_ctx::instance().is_minor_downgrade());
return false;
}
String_type actual_ddl_statement("");
const Object_table_definition *actual_table_def =
object_table->actual_table_definition();
/*
The actual definition may not be present. This will happen during
upgrade if the new DD version adds a new DD table which was not
present in the DD we are upgrading from. This is OK, not an error.
*/
if (actual_table_def == nullptr) return false;
actual_ddl_statement = actual_table_def->get_ddl();
DBUG_ASSERT(!actual_ddl_statement.empty());
return dd::execute_query(thd, actual_ddl_statement);
}
/* purecov: end */
/*
Acquire exclusive meta data locks for the DD schema, tablespace and
table names.
*/
bool acquire_exclusive_mdl(THD *thd) {
// All MDL requests.
MDL_request_list mdl_requests;
// Prepare MDL request for the schema name.
MDL_request schema_request;
MDL_REQUEST_INIT(&schema_request, MDL_key::SCHEMA, MYSQL_SCHEMA_NAME.str, "",
MDL_EXCLUSIVE, MDL_TRANSACTION);
mdl_requests.push_front(&schema_request);
// Prepare MDL request for the tablespace name.
MDL_request tablespace_request;
MDL_REQUEST_INIT(&tablespace_request, MDL_key::TABLESPACE, "",
MYSQL_TABLESPACE_NAME.str, MDL_EXCLUSIVE, MDL_TRANSACTION);
mdl_requests.push_front(&tablespace_request);
// Prepare MDL requests for all tables names.
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
// Skip extraneous tables during minor downgrade.
if ((*it)->entity() == nullptr) continue;
MDL_request *table_request = new (thd->mem_root) MDL_request;
if (table_request == NULL) return true;
MDL_REQUEST_INIT(table_request, MDL_key::TABLE, MYSQL_SCHEMA_NAME.str,
(*it)->entity()->name().c_str(), MDL_EXCLUSIVE,
MDL_TRANSACTION);
mdl_requests.push_front(table_request);
}
// Finally, acquire all the MDL locks.
return (thd->mdl_context.acquire_locks(&mdl_requests,
thd->variables.lock_wait_timeout));
}
/*
Acquire the DD schema, tablespace and table objects. Clone the objects,
reset ID, store persistently, and update the storage adapter.
*/
bool flush_meta_data(THD *thd) {
// Acquire exclusive meta data locks for the relevant DD objects.
if (acquire_exclusive_mdl(thd)) return true;
{
/*
Use a scoped auto releaser to make sure the cached objects are released
before the shared cache is reset.
*/
dd::cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
/*
First, we acquire the DD schema and tablespace and keep them in
local variables. We also clone them, the clones will be used for
updating the ids. We also acquire all the DD table objects to make
sure the shared cache is populated, and we keep the original objects
as well as clones in a vector. The auto releaser will make sure the
objects are not evicted. This must be ensured since we need to make
sure the ids stay consistent across all objects in the shared cache.
*/
const Schema *dd_schema = nullptr;
const Tablespace *dd_tspace = nullptr;
std::vector<const Table_impl *> dd_tables; // Owned by the shared cache.
std::vector<std::unique_ptr<Table_impl>> dd_table_clones;
if (thd->dd_client()->acquire(dd::String_type(MYSQL_SCHEMA_NAME.str),
&dd_schema) ||
thd->dd_client()->acquire(dd::String_type(MYSQL_TABLESPACE_NAME.str),
&dd_tspace))
return dd::end_transaction(thd, true);
std::unique_ptr<Schema_impl> dd_schema_clone(
dynamic_cast<Schema_impl *>(dd_schema->clone()));
std::unique_ptr<Tablespace_impl> dd_tspace_clone(
dynamic_cast<Tablespace_impl *>(dd_tspace->clone()));
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
/*
We add nullptr to the dd_tables vector for abandoned
tables and system tables to have the same number of objects
in the System_tables list, the dd_tables vector and the
dd_table_clones vector.
*/
const dd::Table *dd_table = nullptr;
if ((*it)->property() != System_tables::Types::SYSTEM &&
thd->dd_client()->acquire(MYSQL_SCHEMA_NAME.str,
(*it)->entity()->name(), &dd_table))
return dd::end_transaction(thd, true);
dd_tables.push_back(dynamic_cast<const Table_impl *>(dd_table));
/*
If this is an abandoned table, we can't clone it. Thus, we
push back a nullptr to make sure we have the same number of
elements in the dd_table_clones as in the System_tables.
*/
std::unique_ptr<Table_impl> dd_table_clone;
if (dd_table != nullptr) {
dd_table_clones.push_back(std::unique_ptr<Table_impl>(
dynamic_cast<Table_impl *>(dd_table->clone())));
} else {
dd_table_clones.push_back(nullptr);
}
}
/*
We have now populated the shared cache with the core objects, and kept
clones of all DD objects. The scoped auto releaser makes sure we will
not evict the objects from the shared cache until the auto releaser
exits scope. Thus, within the scope of the auto releaser, we can modify
the contents of the core registry in the storage adapter without risking
that this will interfere with the contents of the shared cache, because
the DD transactions will acquire the core objects from the shared cache.
*/
/*
First, we modify and store the DD schema without changing the cached
copy. We cannot use acquire_for_modification() here, because that
would make the DD sub-transactions (e.g. when calling store()) see a
partially modified set of core objects, where e.g. the mysql
schema object has got its new, real id (from the auto-inc column
in the dd.schemata table), whereas the core DD table objects still
refer to the id that was allocated when creating the scaffolding.
So we first store all the objects persistently, and make sure that
the on-disk data will have correct and consistent ids. When all objects
are stored, we update the contents of the core registry in the
storage adapter to reflect the persisted data. Finally, the shared
cache is reset so that on next acquisition, the DD objects will be
fetched from the core registry in the storage adapter.
*/
/*
We must set the ID to INVALID to make the object get a fresh ID from
the auto inc ID column.
*/
dd_schema_clone->set_id(INVALID_OBJECT_ID);
dd_tspace_clone->set_id(INVALID_OBJECT_ID);
if (dd::cache::Storage_adapter::instance()->store(
thd, static_cast<Schema *>(dd_schema_clone.get())) ||
dd::cache::Storage_adapter::instance()->store(
thd, static_cast<Tablespace *>(dd_tspace_clone.get())))
return dd::end_transaction(thd, true);
/*
Now, the DD schema and DD tablespace are stored persistently. We will
not update the core registry until after we have stored all DD tables.
At that point, we can update all the core registry objects in one go
and avoid using a partially update core registry for e.g. object
acquisition.
*/
std::vector<std::unique_ptr<Table_impl>>::iterator clone_it =
dd_table_clones.begin();
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end() &&
clone_it != dd_table_clones.end();
++it, ++clone_it) {
// Skip abandoned tables and system tables.
if ((*clone_it) == nullptr ||
(*it)->property() == System_tables::Types::SYSTEM)
continue;
DBUG_ASSERT((*it)->entity()->name() == (*clone_it)->name());
// We must set the ID to INVALID to let the object get an auto inc ID.
(*clone_it)->set_id(INVALID_OBJECT_ID);
/*
Change the schema and tablespace id to match the ids of the
persisted objects. Note that this means the persisted DD table
objects will have consistent IDs, but the IDs in the objects in
the core registry will not be updated yet.
*/
(*clone_it)->set_schema_id(dd_schema_clone->id());
(*clone_it)->set_tablespace_id(dd_tspace_clone->id());
if (dd::cache::Storage_adapter::instance()->store(
thd, static_cast<Table *>((*clone_it).get())))
return dd::end_transaction(thd, true);
}
/*
Update and store the predefined tablespace objects. The DD tablespace
has already been stored above, so we iterate only over the tablespaces
of type PREDEFINED_DDSE.
*/
for (System_tablespaces::Const_iterator it =
System_tablespaces::instance()->begin(
System_tablespaces::Types::PREDEFINED_DDSE);
it != System_tablespaces::instance()->end();
it = System_tablespaces::instance()->next(
it, System_tablespaces::Types::PREDEFINED_DDSE)) {
const dd::Tablespace *tspace = nullptr;
if (thd->dd_client()->acquire((*it)->key().second, &tspace))
return dd::end_transaction(thd, true);
std::unique_ptr<Tablespace_impl> tspace_clone(
dynamic_cast<Tablespace_impl *>(tspace->clone()));
// We must set the ID to INVALID to enable storing the object.
tspace_clone->set_id(INVALID_OBJECT_ID);
if (dd::cache::Storage_adapter::instance()->store(
thd, static_cast<Tablespace *>(tspace_clone.get())))
return dd::end_transaction(thd, true);
/*
Only the DD tablespace is needed to handle cache misses, so we can
just drop the predefined tablespaces from the core registry now that
it has been persisted.
*/
dd::cache::Storage_adapter::instance()->core_drop(thd, tspace);
}
/*
Now, the DD schema and tablespace as well as the DD tables have been
persisted. The last thing we do before resetting the shared cache is
to update the contents of the core registry to match the persisted
objects. First, we update the core registry with the persisted DD
schema and tablespace.
*/
dd::cache::Storage_adapter::instance()->core_drop(thd, dd_schema);
dd::cache::Storage_adapter::instance()->core_store(
thd, static_cast<Schema *>(dd_schema_clone.get()));
dd::cache::Storage_adapter::instance()->core_drop(thd, dd_tspace);
dd::cache::Storage_adapter::instance()->core_store(
thd, static_cast<Tablespace *>(dd_tspace_clone.get()));
// Make sure the IDs after storing are as expected.
DBUG_ASSERT(dd_schema_clone->id() == 1);
DBUG_ASSERT(dd_tspace_clone->id() == 1);
/*
Finally, we update the core registry of the DD tables. This must be
done in two loops to avoid issues related to overlapping ID sequences.
*/
std::vector<const Table_impl *>::const_iterator table_it =
dd_tables.begin();
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end() && table_it != dd_tables.end();
++it, ++table_it) {
// Skip abandoned tables and system tables.
if ((*table_it) == nullptr ||
(*it)->property() == System_tables::Types::SYSTEM)
continue;
DBUG_ASSERT((*it)->entity()->name() == (*table_it)->name());
dd::cache::Storage_adapter::instance()->core_drop(
thd, static_cast<const Table *>(*table_it));
}
clone_it = dd_table_clones.begin();
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end() &&
clone_it != dd_table_clones.end();
++it, ++clone_it) {
// Skip abandoned tables and system tables.
if ((*clone_it) == nullptr ||
(*it)->property() == System_tables::Types::SYSTEM)
continue;
if ((*it)->property() == System_tables::Types::CORE) {
DBUG_ASSERT((*it)->entity()->name() == (*clone_it)->name());
dd::cache::Storage_adapter::instance()->core_store(
thd, static_cast<Table *>((*clone_it).get()));
}
}
}
/*
Now, the auto releaser has released the objects, and we can go ahead and
reset the shared cache.
*/
dd::cache::Shared_dictionary_cache::instance()->reset(true);
if (dd::end_transaction(thd, false)) {
return true;
}
/*
Use a scoped auto releaser to make sure the objects cached for SDI
writing, FK parent information reload, and DD property storage are
released.
*/
dd::cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
// Acquire the DD tablespace and write SDI
const Tablespace *dd_tspace = nullptr;
if (thd->dd_client()->acquire(dd::String_type(MYSQL_TABLESPACE_NAME.str),
&dd_tspace) ||
dd::sdi::store(thd, dd_tspace)) {
return dd::end_transaction(thd, true);
}
// Acquire the DD schema and write SDI
const Schema *dd_schema = nullptr;
if (thd->dd_client()->acquire(dd::String_type(MYSQL_SCHEMA_NAME.str),
&dd_schema) ||
dd::sdi::store(thd, dd_schema)) {
return dd::end_transaction(thd, true);
}
/*
Acquire the DD table objects and write SDI for them. Also sync from
the DD tables in order to get the FK parent information reloaded.
*/
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
// Skip system tables.
if ((*it)->property() == System_tables::Types::SYSTEM) continue;
const dd::Table *dd_table = nullptr;
if (thd->dd_client()->acquire(MYSQL_SCHEMA_NAME.str,
(*it)->entity()->name(), &dd_table)) {
return dd::end_transaction(thd, true);
}
// Skip abandoned tables.
if (dd_table == nullptr) continue;
/*
Make sure the registry of the core DD objects is updated with an
object read from the DD tables, with updated FK parent information.
Store the object to make sure SDI is written.
*/
Abstract_table::Name_key table_key;
Abstract_table::update_name_key(&table_key, dd_schema->id(),
dd_table->name());
const dd::Abstract_table *persisted_dd_table = nullptr;
if (dd::cache::Storage_adapter::instance()->get(
thd, table_key, ISO_READ_COMMITTED, true, &persisted_dd_table) ||
persisted_dd_table == nullptr ||
dd::sdi::store(thd, dynamic_cast<const Table *>(persisted_dd_table))) {
if (persisted_dd_table != nullptr) delete persisted_dd_table;
return dd::end_transaction(thd, true);
}
if ((*it)->property() == System_tables::Types::CORE) {
dd::cache::Storage_adapter::instance()->core_drop(thd, dd_table);
dd::cache::Storage_adapter::instance()->core_store(
thd, dynamic_cast<Table *>(
const_cast<Abstract_table *>(persisted_dd_table)));
}
if (persisted_dd_table != nullptr) delete persisted_dd_table;
}
bootstrap::DD_bootstrap_ctx::instance().set_stage(bootstrap::Stage::SYNCED);
return dd::end_transaction(thd, false);
}
// Insert additional data into the DD tables.
bool populate_tables(THD *thd) {
// Iterate over DD tables, populate tables.
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
// Skip system tables.
if ((*it)->property() == System_tables::Types::SYSTEM) continue;
// Retrieve list of SQL statements to execute.
const Object_table_definition *table_def =
(*it)->entity()->target_table_definition();
// Skip abandoned tables.
if (table_def == nullptr) continue;
bool error = false;
std::vector<dd::String_type> stmt = table_def->get_dml();
for (std::vector<dd::String_type>::iterator stmt_it = stmt.begin();
stmt_it != stmt.end() && !error; ++stmt_it)
error = dd::execute_query(thd, *stmt_it);
// Commit the statement based population.
if (dd::end_transaction(thd, error)) return true;
// If no error, call the low level table population method, and commit it.
error = (*it)->entity()->populate(thd);
if (dd::end_transaction(thd, error)) return true;
}
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::POPULATED);
return false;
}
// Re-populate character sets and collations upon normal restart.
bool repopulate_charsets_and_collations(THD *thd) {
/*
We must check if the DDSE is started in a way that makes the DD
read only. For now, we only support InnoDB as SE for the DD. The call
to retrieve the handlerton for the DDSE should be replaced by a more
generic mechanism.
*/
handlerton *ddse = ha_resolve_by_legacy_type(thd, DB_TYPE_INNODB);
if (ddse->is_dict_readonly && ddse->is_dict_readonly()) {
LogErr(WARNING_LEVEL, ER_DD_NO_WRITES_NO_REPOPULATION, "InnoDB", " ");
return false;
}
/*
Otherwise, turn off FK checks, re-populate and commit.
The FK checks must be turned off since the collations and
character sets reference each other.
*/
bool error = dd::execute_query(thd, "SET FOREIGN_KEY_CHECKS= 0") ||
tables::Collations::instance().populate(thd) ||
tables::Character_sets::instance().populate(thd);
/*
We must commit the re-population before executing a new query, which
expects the transaction to be empty, and finally, turn FK checks back on.
*/
error |= dd::end_transaction(thd, error);
error |= dd::execute_query(thd, "SET FOREIGN_KEY_CHECKS= 1");
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::POPULATED);
return error;
}
/*
Verify that the storage adapter contains the core DD objects and
nothing else.
*/
bool verify_contents(THD *thd) {
// Verify that the DD schema is present, and that its id == 1.
Schema::Name_key schema_key;
Schema::update_name_key(&schema_key, MYSQL_SCHEMA_NAME.str);
Object_id dd_schema_id =
cache::Storage_adapter::instance()->core_get_id<Schema>(schema_key);
DBUG_ASSERT(dd_schema_id == MYSQL_SCHEMA_DD_ID);
if (dd_schema_id == INVALID_OBJECT_ID) {
LogErr(ERROR_LEVEL, ER_DD_SCHEMA_NOT_FOUND, MYSQL_SCHEMA_NAME.str);
return dd::end_transaction(thd, true);
}
DBUG_ASSERT(cache::Storage_adapter::instance()->core_size<Schema>() == 1);
// Verify that the core DD tables are present.
#ifndef DBUG_OFF
size_t n_core_tables = 0;
#endif
for (System_tables::Const_iterator it =
System_tables::instance()->begin(System_tables::Types::CORE);
it != System_tables::instance()->end();
it = System_tables::instance()->next(it, System_tables::Types::CORE)) {
// Skip extraneous tables for minor downgrade.
if ((*it)->entity() == nullptr) continue;
#ifndef DBUG_OFF
n_core_tables++;
#endif
Table::Name_key table_key;
Table::update_name_key(&table_key, dd_schema_id, (*it)->entity()->name());
Object_id dd_table_id =
cache::Storage_adapter::instance()->core_get_id<Table>(table_key);
DBUG_ASSERT(dd_table_id != INVALID_OBJECT_ID);
if (dd_table_id == INVALID_OBJECT_ID) {
LogErr(ERROR_LEVEL, ER_DD_TABLE_NOT_FOUND,
(*it)->entity()->name().c_str());
return dd::end_transaction(thd, true);
}
}
DBUG_ASSERT(cache::Storage_adapter::instance()->core_size<Abstract_table>() ==
n_core_tables);
// Verify that the dictionary tablespace is present and that its id == 1.
Tablespace::Name_key tspace_key;
Tablespace::update_name_key(&tspace_key, MYSQL_TABLESPACE_NAME.str);
Object_id dd_tspace_id =
cache::Storage_adapter::instance()->core_get_id<Tablespace>(tspace_key);
DBUG_ASSERT(dd_tspace_id == MYSQL_TABLESPACE_DD_ID);
if (dd_tspace_id == INVALID_OBJECT_ID) {
LogErr(ERROR_LEVEL, ER_DD_TABLESPACE_NOT_FOUND, MYSQL_TABLESPACE_NAME.str);
return dd::end_transaction(thd, true);
}
DBUG_ASSERT(cache::Storage_adapter::instance()->core_size<Tablespace>() == 1);
return dd::end_transaction(thd, false);
}
} // namespace
///////////////////////////////////////////////////////////////////////////
namespace dd {
namespace bootstrap {
/*
Do the necessary DD-related initialization in the DDSE, and get the
predefined tables and tablespaces.
*/
bool DDSE_dict_init(THD *thd, dict_init_mode_t dict_init_mode, uint version) {
handlerton *ddse = ha_resolve_by_legacy_type(thd, DB_TYPE_INNODB);
/*
The lists with element wrappers are mem root allocated. The wrapped
instances are allocated dynamically in the DDSE. These instances will be
owned by the System_tables registry by the end of this function.
*/
List<const Object_table> ddse_tables;
List<const Plugin_tablespace> ddse_tablespaces;
if (ddse->ddse_dict_init == nullptr ||
ddse->ddse_dict_init(dict_init_mode, version, &ddse_tables,
&ddse_tablespaces))
return true;
/*
Iterate over the table definitions and add them to the System_tables
registry. The Object_table instances will later be used to execute
CREATE TABLE statements to actually create the tables.
If Object_table::is_hidden(), then we add the tables as type DDSE_PRIVATE
(not available neither for DDL nor DML), otherwise, we add them as type
DDSE_PROTECTED (available for DML, not for DDL).
*/
List_iterator<const Object_table> table_it(ddse_tables);
const Object_table *ddse_table = nullptr;
while ((ddse_table = table_it++)) {
System_tables::Types table_type = System_tables::Types::DDSE_PROTECTED;
if (ddse_table->is_hidden()) {
table_type = System_tables::Types::DDSE_PRIVATE;
}
System_tables::instance()->add(MYSQL_SCHEMA_NAME.str, ddse_table->name(),
table_type, ddse_table);
}
/*
Get the server version number from the DD tablespace header and verify
that we are allowed to upgrade from that version. The error handling is done
after adding the ddse tables into the system registry to avoid memory leaks.
*/
if (!opt_initialize) {
uint server_version = 0;
if (ddse->dict_get_server_version == nullptr ||
ddse->dict_get_server_version(&server_version)) {
LogErr(ERROR_LEVEL, ER_CANNOT_GET_SERVER_VERSION_FROM_TABLESPACE_HEADER);
return true;
}
if (server_version != MYSQL_VERSION_ID) {
if (opt_upgrade_mode == UPGRADE_NONE) {
LogErr(ERROR_LEVEL, ER_SERVER_UPGRADE_OFF);
return true;
}
if (!DD_bootstrap_ctx::instance().supported_server_version(
server_version)) {
LogErr(ERROR_LEVEL, ER_SERVER_UPGRADE_VERSION_NOT_SUPPORTED,
server_version);
return true;
}
}
}
/*
At this point, the Systen_tables registry contains the INERT DD tables,
and the DDSE tables. Before we continue, we must add the remaining
DD tables.
*/
System_tables::instance()->add_remaining_dd_tables();
/*
Iterate over the tablespace definitions, add the names and the
tablespace meta data to the System_tablespaces registry. The
meta data will be used later to create dd::Tablespace objects.
The Plugin_tablespace instances are owned by the DDSE.
*/
List_iterator<const Plugin_tablespace> tablespace_it(ddse_tablespaces);
const Plugin_tablespace *tablespace = nullptr;
while ((tablespace = tablespace_it++)) {
// Add the name and the object instance to the registry with the
// appropriate property.
if (my_strcasecmp(system_charset_info, MYSQL_TABLESPACE_NAME.str,
tablespace->get_name()) == 0)
System_tablespaces::instance()->add(
tablespace->get_name(), System_tablespaces::Types::DD, tablespace);
else
System_tablespaces::instance()->add(
tablespace->get_name(), System_tablespaces::Types::PREDEFINED_DDSE,
tablespace);
}
return false;
}
// Initialize the data dictionary.
bool initialize_dictionary(THD *thd, bool is_dd_upgrade_57,
Dictionary_impl *d) {
if (is_dd_upgrade_57)
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::STARTED);
store_predefined_tablespace_metadata(thd);
if (create_dd_schema(thd) || initialize_dd_properties(thd) ||
create_tables(thd, nullptr))
return true;
if (is_dd_upgrade_57) {
// Add status to mark creation of dictionary in InnoDB.
// Till this step, no new undo log is created by InnoDB.
if (upgrade_57::Upgrade_status().update(
upgrade_57::Upgrade_status::enum_stage::DICT_TABLES_CREATED))
return true;
}
DBUG_EXECUTE_IF(
"dd_upgrade_stage_2", if (is_dd_upgrade_57) {
/*
Server will crash will upgrading 5.7 data directory.
This will leave server is an inconsistent state.
File tracking upgrade will have Stage 2 written in it.
Next restart of server on same data directory should
revert all changes done by upgrade and data directory
should be reusable by 5.7 server.
*/
DBUG_SUICIDE();
});
if (DDSE_dict_recover(thd, DICT_RECOVERY_INITIALIZE_SERVER,
d->get_target_dd_version()) ||
flush_meta_data(thd) ||
DDSE_dict_recover(thd, DICT_RECOVERY_INITIALIZE_TABLESPACES,
d->get_target_dd_version()) ||
populate_tables(thd) ||
update_properties(thd, nullptr, nullptr,
String_type(MYSQL_SCHEMA_NAME.str)) ||
verify_contents(thd) | update_versions(thd, is_dd_upgrade_57))
return true;
bootstrap::DD_bootstrap_ctx::instance().set_stage(bootstrap::Stage::FINISHED);
return false;
}
// First time server start and initialization of the data dictionary.
bool initialize(THD *thd) {
bootstrap::DD_bootstrap_ctx::instance().set_stage(bootstrap::Stage::STARTED);
/*
Set tx_read_only to false to allow installing DD tables even
if the server is started with --transaction-read-only=true.
*/
thd->variables.transaction_read_only = false;
thd->tx_read_only = false;
Disable_autocommit_guard autocommit_guard(thd);
Dictionary_impl *d = dd::Dictionary_impl::instance();
DBUG_ASSERT(d);
cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
/*
Each step in the install process below is committed independently,
either implicitly (for e.g. "CREATE TABLE") or explicitly (for the
operations in the "populate()" methods). Thus, there is no need to
commit explicitly here.
*/
if (DDSE_dict_init(thd, DICT_INIT_CREATE_FILES, d->get_target_dd_version()) ||
initialize_dictionary(thd, false, d))
return true;
DBUG_ASSERT(d->get_target_dd_version() == d->get_actual_dd_version(thd));
LogErr(INFORMATION_LEVEL, ER_DD_VERSION_INSTALLED,
d->get_target_dd_version());
return false;
}
// Normal server restart.
bool restart(THD *thd) {
bootstrap::DD_bootstrap_ctx::instance().set_stage(bootstrap::Stage::STARTED);
/*
Set tx_read_only to false to allow installing DD tables even
if the server is started with --transaction-read-only=true.
*/
thd->variables.transaction_read_only = false;
thd->tx_read_only = false;
// Set explicit_defaults_for_timestamp variable for dictionary creation
thd->variables.explicit_defaults_for_timestamp = true;
Disable_autocommit_guard autocommit_guard(thd);
Dictionary_impl *d = dd::Dictionary_impl::instance();
DBUG_ASSERT(d);
cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
store_predefined_tablespace_metadata(thd);
if (create_dd_schema(thd) || initialize_dd_properties(thd) ||
create_tables(thd, nullptr) || sync_meta_data(thd) ||
DDSE_dict_recover(thd, DICT_RECOVERY_RESTART_SERVER,
d->get_actual_dd_version(thd)) ||
upgrade::do_server_upgrade_checks(thd) || upgrade::upgrade_tables(thd) ||
repopulate_charsets_and_collations(thd) || verify_contents(thd) ||
update_versions(thd, false)) {
return true;
}
bootstrap::DD_bootstrap_ctx::instance().set_stage(bootstrap::Stage::FINISHED);
LogErr(INFORMATION_LEVEL, ER_DD_VERSION_FOUND, d->get_actual_dd_version(thd));
return false;
}
// Initialize dictionary in case of server restart.
void recover_innodb_upon_upgrade(THD *thd) {
Dictionary_impl *d = dd::Dictionary_impl::instance();
store_predefined_tablespace_metadata(thd);
// RAII to handle error in execution of CREATE TABLE.
Key_length_error_handler key_error_handler;
/*
Ignore ER_TOO_LONG_KEY for dictionary tables during restart.
Do not print the error in error log as we are creating only the
cached objects and not physical tables.
TODO: Workaround due to bug#20629014. Remove when the bug is fixed.
*/
thd->push_internal_handler(&key_error_handler);
if (create_dd_schema(thd) || initialize_dd_properties(thd) ||
create_tables(thd, nullptr) ||
DDSE_dict_recover(thd, DICT_RECOVERY_RESTART_SERVER,
d->get_actual_dd_version(thd))) {
// Error is not be handled in this case as we are on cleanup code path.
LogErr(WARNING_LEVEL, ER_DD_INIT_UPGRADE_FAILED);
}
thd->pop_internal_handler();
return;
}
bool setup_dd_objects_and_collations(THD *thd) {
// Continue with server startup.
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::CREATED_TABLES);
/*
Set tx_read_only to false to allow installing DD tables even
if the server is started with --transaction-read-only=true.
*/
thd->variables.transaction_read_only = false;
thd->tx_read_only = false;
Disable_autocommit_guard autocommit_guard(thd);
Dictionary_impl *d = dd::Dictionary_impl::instance();
DBUG_ASSERT(d);
DBUG_ASSERT(d->get_target_dd_version() == d->get_actual_dd_version(thd));
/*
In this context, we initialize the target tables directly since this
is a restart based on a pre-transactional-DD server, so ordinary
upgrade does not need to be considered.
*/
if (sync_meta_data(thd) || repopulate_charsets_and_collations(thd) ||
verify_contents(thd) || update_versions(thd, false)) {
return true;
}
bootstrap::DD_bootstrap_ctx::instance().set_stage(bootstrap::Stage::FINISHED);
LogErr(INFORMATION_LEVEL, ER_DD_VERSION_FOUND, d->get_actual_dd_version(thd));
return false;
}
} // namespace bootstrap
void store_predefined_tablespace_metadata(THD *thd) {
/*
Create dd::Tablespace objects and store them (which will add their meta
data to the storage adapter registry of DD entities). The tablespaces
are already created physically in the DDSE, so we only need to create
the corresponding meta data.
*/
for (System_tablespaces::Const_iterator it =
System_tablespaces::instance()->begin();
it != System_tablespaces::instance()->end(); ++it) {
const Plugin_tablespace *tablespace_def = (*it)->entity();
// Create the dd::Tablespace object.
std::unique_ptr<Tablespace> tablespace(dd::create_object<Tablespace>());
tablespace->set_name(tablespace_def->get_name());
tablespace->set_options(tablespace_def->get_options());
tablespace->set_se_private_data(tablespace_def->get_se_private_data());
tablespace->set_engine(tablespace_def->get_engine());
// Loop over the tablespace files, create dd::Tablespace_file objects.
List<const Plugin_tablespace::Plugin_tablespace_file> files =
tablespace_def->get_files();
List_iterator<const Plugin_tablespace::Plugin_tablespace_file> file_it(
files);
const Plugin_tablespace::Plugin_tablespace_file *file = NULL;
while ((file = file_it++)) {
Tablespace_file *space_file = tablespace->add_file();
space_file->set_filename(file->get_name());
space_file->set_se_private_data(file->get_se_private_data());
}
// All the predefined tablespace are unencrypted (atleast for now).
tablespace->options().set("encryption", "N");
/*
Here, we just want to populate the core registry in the storage
adapter. We do not want to have the object registered in the
uncommitted registry, this will only add complexity to the
DD cache usage during bootstrap. Thus, we call the storage adapter
directly instead of going through the dictionary client.
*/
dd::cache::Storage_adapter::instance()->store(thd, tablespace.get());
}
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::CREATED_TABLESPACES);
}
bool create_dd_schema(THD *thd) {
return dd::execute_query(thd,
dd::String_type("CREATE SCHEMA ") +
dd::String_type(MYSQL_SCHEMA_NAME.str) +
dd::String_type(" DEFAULT COLLATE ") +
dd::String_type(default_charset_info->name)) ||
dd::execute_query(thd, dd::String_type("USE ") +
dd::String_type(MYSQL_SCHEMA_NAME.str));
}
bool initialize_dd_properties(THD *thd) {
// Create the dd_properties table.
const Object_table_definition *dd_properties_def =
dd::tables::DD_properties::instance().target_table_definition();
if (dd::execute_query(thd, dd_properties_def->get_ddl())) return true;
/*
We can now decide which version number we will use for the DD, and
initialize the DD_bootstrap_ctx with the relevant version number.
*/
uint actual_version = dd::DD_VERSION;
uint actual_server_version = MYSQL_VERSION_ID;
uint upgraded_server_version = MYSQL_VERSION_ID;
bootstrap::DD_bootstrap_ctx::instance().set_actual_dd_version(actual_version);
bootstrap::DD_bootstrap_ctx::instance().set_upgraded_server_version(
actual_server_version);
Minor_upgrade_ctx::instance()->set_extra_mvu_version(
Minor_upgrade_ctx::instance()->get_target_extra_mvu_version());
if (!opt_initialize) {
bool exists = false;
bool exists_server = false;
bool exists_upgraded_version = false;
// Check 'DD_version' too in order to catch an upgrade from 8.0.3.
if (dd::tables::DD_properties::instance().get(thd, "DD_VERSION",
&actual_version, &exists) ||
!exists) {
LogErr(ERROR_LEVEL, ER_DD_NO_VERSION_FOUND);
return true;
}
/* purecov: begin inspected */
if (actual_version != dd::DD_VERSION) {
bootstrap::DD_bootstrap_ctx::instance().set_actual_dd_version(
actual_version);
if (opt_no_dd_upgrade) {
push_deprecated_warn(thd, "--no-dd-upgrade", "--upgrade=NONE");
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_OFF);
return true;
}
if (!bootstrap::DD_bootstrap_ctx::instance().supported_dd_version()) {
/*
If we are attempting on minor downgrade, make sure this is
supported.
*/
if (!bootstrap::DD_bootstrap_ctx::instance().is_minor_downgrade()) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_VERSION_NOT_SUPPORTED,
actual_version);
return true;
}
uint minor_downgrade_threshold = 0;
if (dd::tables::DD_properties::instance().get(
thd, "MINOR_DOWNGRADE_THRESHOLD", &minor_downgrade_threshold,
&exists) ||
!exists || minor_downgrade_threshold > dd::DD_VERSION) {
LogErr(ERROR_LEVEL, ER_DD_MINOR_DOWNGRADE_VERSION_NOT_SUPPORTED,
actual_version);
return true;
}
}
}
/* purecov: end */
if (dd::tables::DD_properties::instance().get(
thd, "MYSQLD_VERSION", &actual_server_version, &exists_server) ||
!exists_server)
return true;
if (dd::tables::DD_properties::instance().get(
thd, "MYSQLD_VERSION_UPGRADED", &upgraded_server_version,
&exists_upgraded_version) ||
!exists_upgraded_version)
upgraded_server_version = actual_server_version;
bootstrap::DD_bootstrap_ctx::instance().set_upgraded_server_version(
upgraded_server_version);
Minor_upgrade_ctx::instance()->retrieve_and_set_extra_mvu_version(thd);
if (DBUG_EVALUATE_IF("simulate_mysql_upgrade_skip_pending", true,
actual_server_version != upgraded_server_version &&
actual_server_version != MYSQL_VERSION_ID)) {
LogErr(ERROR_LEVEL, ER_SERVER_UPGRADE_PENDING, MYSQL_VERSION_ID,
upgraded_server_version);
return true;
}
if (upgraded_server_version != MYSQL_VERSION_ID) {
/*
This check is also done in DDSE_dict_init() based on the version
number from the DD tablespace header. Here, we repeat the check,
this time based on the server version number stored in the DD
table 'dd_properties'. The two checks should give the same result,
so this check should never fail; hence, the debug assert.
*/
if (!bootstrap::DD_bootstrap_ctx::instance().supported_server_version()) {
LogErr(ERROR_LEVEL, ER_SERVER_UPGRADE_VERSION_NOT_SUPPORTED,
actual_server_version);
DBUG_ASSERT(false);
return true;
}
}
/*
Reject restarting with a changed LCTN setting, since the collation
for LCTN-dependent columns is decided during server initialization.
*/
uint actual_lctn = 0;
exists = false;
if (dd::tables::DD_properties::instance().get(thd, "LCTN", &actual_lctn,
&exists) ||
!exists) {
LogErr(WARNING_LEVEL, ER_LCTN_NOT_FOUND, lower_case_table_names);
} else if (actual_lctn != lower_case_table_names) {
LogErr(ERROR_LEVEL, ER_LCTN_CHANGED, lower_case_table_names, actual_lctn);
return true;
}
}
if (bootstrap::DD_bootstrap_ctx::instance().is_initialize())
LogErr(INFORMATION_LEVEL, ER_DD_INITIALIZE, dd::DD_VERSION);
else if (bootstrap::DD_bootstrap_ctx::instance().is_restart())
LogErr(INFORMATION_LEVEL, ER_DD_RESTART, dd::DD_VERSION);
else if (bootstrap::DD_bootstrap_ctx::instance().is_minor_downgrade())
LogErr(INFORMATION_LEVEL, ER_DD_MINOR_DOWNGRADE, actual_version,
dd::DD_VERSION);
else {
/*
If none of the above, then this must be DD upgrade or server
upgrade, or both.
*/
if (bootstrap::DD_bootstrap_ctx::instance().is_dd_upgrade()) {
LogErr(SYSTEM_LEVEL, ER_DD_UPGRADE, actual_version, dd::DD_VERSION);
log_sink_buffer_check_timeout();
sysd::notify("STATUS=Data Dictionary upgrade in progress\n");
}
if (bootstrap::DD_bootstrap_ctx::instance().is_server_upgrade()) {
// This condition is hit only if upgrade has been skipped before
if (opt_upgrade_mode == UPGRADE_NONE) {
LogErr(ERROR_LEVEL, ER_SERVER_UPGRADE_OFF);
return true;
}
LogErr(INFORMATION_LEVEL, ER_SERVER_UPGRADE_FROM_VERSION,
upgraded_server_version, MYSQL_VERSION_ID);
LogErr(INFORMATION_LEVEL, ER_SERVER_UPGRADE_FROM_VERSION,
Minor_upgrade_ctx::instance()->get_extra_mvu_version(),
Minor_upgrade_ctx::instance()->get_target_extra_mvu_version());
}
DBUG_ASSERT(bootstrap::DD_bootstrap_ctx::instance().is_dd_upgrade() ||
bootstrap::DD_bootstrap_ctx::instance().is_server_upgrade());
}
/*
Unless this is initialization or restart, we must update the
System_tables registry with the information from the 'dd_properties'
regarding the actual DD tables.
*/
if (!bootstrap::DD_bootstrap_ctx::instance().is_initialize() &&
bootstrap::DD_bootstrap_ctx::instance().is_dd_upgrade() &&
update_system_tables(thd)) {
return true;
}
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::FETCHED_PROPERTIES);
return false;
}
bool is_non_inert_dd_or_ddse_table(System_tables::Types table_type) {
return table_type == System_tables::Types::CORE ||
table_type == System_tables::Types::SECOND ||
table_type == System_tables::Types::DDSE_PRIVATE ||
table_type == System_tables::Types::DDSE_PROTECTED;
}
bool create_tables(THD *thd, const std::set<String_type> *create_set) {
// Turn off FK checks, this is needed since we have cyclic FKs.
if (dd::execute_query(thd, "SET FOREIGN_KEY_CHECKS= 0")) return true;
/*
Decide whether we should create actual or target tables. For plain
restart and initialize, we create the target tables. For the second
table creation stage during upgrade, we also create target tables.
So we create the actual tables only during the first table creation
stage for upgrade, and for minor downgrade.
*/
bool create_target_tables = true;
if (bootstrap::DD_bootstrap_ctx::instance().get_stage() ==
bootstrap::Stage::FETCHED_PROPERTIES &&
(bootstrap::DD_bootstrap_ctx::instance().is_dd_upgrade() ||
bootstrap::DD_bootstrap_ctx::instance().is_minor_downgrade()))
create_target_tables = false;
/*
Iterate over DD tables and create the tables. Note that we do not iterate
over INERT tables here, there is currently only one INERT table (the
'dd_properties'), and it is created in 'initialize_dd_properties' in
order to get hold of e.g. version information.
*/
bool error = false;
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end() && !error; ++it) {
if (is_non_inert_dd_or_ddse_table((*it)->property())) {
/*
If a create set is submitted, create only the target tables that
are in the create set.
*/
if (create_set == nullptr ||
create_set->find((*it)->entity()->name()) != create_set->end()) {
/*
Use the actual or target definition to create the table depending
on the context.
*/
if (create_target_tables)
error = create_target_table(thd, (*it)->entity());
else
error = create_actual_table(thd, (*it)->entity());
}
}
}
// Turn FK checks back on.
if (error || dd::execute_query(thd, "SET FOREIGN_KEY_CHECKS= 1")) return true;
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::CREATED_TABLES);
return false;
}
bool sync_meta_data(THD *thd) {
// Acquire exclusive meta data locks for the relevant DD objects.
if (acquire_exclusive_mdl(thd)) return true;
{
/*
Use a scoped auto releaser to make sure the cached objects are released
before the shared cache is reset.
*/
dd::cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
/*
First, we acquire the DD schema and tablespace and keep them in
local variables. The DD table objects are acquired and put into
a vector. We also get hold of the corresponding persisted objects.
In this way, we make sure the shared cache is populated. The auto
releaser will make sure the objects are not evicted. This must be
ensured since we need to make sure the ids stay consistent across
all objects in the shared cache.
*/
const Schema *dd_schema = nullptr;
const Tablespace *dd_tspace = nullptr;
if (thd->dd_client()->acquire(dd::String_type(MYSQL_SCHEMA_NAME.str),
&dd_schema) ||
thd->dd_client()->acquire(dd::String_type(MYSQL_TABLESPACE_NAME.str),
&dd_tspace))
return dd::end_transaction(thd, true);
std::vector<Table *> dd_tables; // Owned by the shared cache.
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
// Skip extraneous tables during minor downgrade.
if ((*it)->entity() == nullptr) continue;
const dd::Table *dd_table = nullptr;
if (thd->dd_client()->acquire(MYSQL_SCHEMA_NAME.str,
(*it)->entity()->name(), &dd_table))
return dd::end_transaction(thd, true);
dd_tables.push_back(const_cast<Table *>(dd_table));
}
// Get the persisted DD schema and tablespace.
Schema::Name_key schema_key;
dd_schema->update_name_key(&schema_key);
const Schema *tmp_schema = nullptr;
Tablespace::Name_key tspace_key;
dd_tspace->update_name_key(&tspace_key);
const Tablespace *tmp_tspace = nullptr;
if (dd::cache::Storage_adapter::instance()->get(
thd, schema_key, ISO_READ_COMMITTED, true, &tmp_schema) ||
dd::cache::Storage_adapter::instance()->get(
thd, tspace_key, ISO_READ_COMMITTED, true, &tmp_tspace))
return dd::end_transaction(thd, true);
DBUG_ASSERT(tmp_schema != nullptr && tmp_tspace != nullptr);
std::unique_ptr<Schema> persisted_dd_schema(
const_cast<Schema *>(tmp_schema));
std::unique_ptr<Tablespace> persisted_dd_tspace(
const_cast<Tablespace *>(tmp_tspace));
// If the persisted meta data indicates that the DD tablespace is
// encrypted, then we record this fact to make sure the DDL statements
// that are genereated during e.g. upgrade will have the correct
// encryption option.
String_type encryption("");
Object_table_definition_impl::set_dd_tablespace_encrypted(
persisted_dd_tspace->options().exists("encryption") &&
!persisted_dd_tspace->options().get("encryption", &encryption) &&
encryption == "Y");
// Get the persisted DD table objects into a vector.
std::vector<std::unique_ptr<Table_impl>> persisted_dd_tables;
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
// Skip extraneous tables during minor downgrade.
if ((*it)->entity() == nullptr) continue;
const dd::Abstract_table *dd_table = nullptr;
dd::Abstract_table::Name_key table_key;
Abstract_table::update_name_key(&table_key, persisted_dd_schema->id(),
(*it)->entity()->name());
if (dd::cache::Storage_adapter::instance()->get(
thd, table_key, ISO_READ_COMMITTED, true, &dd_table))
return dd::end_transaction(thd, true);
std::unique_ptr<Table_impl> persisted_dd_table(
dynamic_cast<Table_impl *>(const_cast<Abstract_table *>(dd_table)));
persisted_dd_tables.push_back(std::move(persisted_dd_table));
}
// Drop the tablespaces with type PREDEFINED_DDSE from the storage adapter.
for (System_tablespaces::Const_iterator it =
System_tablespaces::instance()->begin(
System_tablespaces::Types::PREDEFINED_DDSE);
it != System_tablespaces::instance()->end();
it = System_tablespaces::instance()->next(
it, System_tablespaces::Types::PREDEFINED_DDSE)) {
const Tablespace *tspace = nullptr;
if (thd->dd_client()->acquire((*it)->entity()->get_name(), &tspace))
return dd::end_transaction(thd, true);
dd::cache::Storage_adapter::instance()->core_drop(thd, tspace);
}
/*
We have now populated the shared cache with the core objects. The
scoped auto releaser makes sure we will not evict the objects from
the shared cache until the auto releaser exits scope. Thus, within
the scope of the auto releaser, we can modify the contents of the
core registry in the storage adapter without risking that this will
interfere with the contents of the shared cache, because the DD
transactions will acquire the core objects from the shared cache.
*/
/*
We have also read the DD schema and tablespace as well as the DD
tables from persistent storage. The last thing we do before resetting
the shared cache is to update the contents of the core registry to
match the persisted objects. First, we update the core registry with
the persisted DD schema and tablespace.
*/
dd::cache::Storage_adapter::instance()->core_drop(thd, dd_schema);
dd::cache::Storage_adapter::instance()->core_store(
thd, persisted_dd_schema.get());
dd::cache::Storage_adapter::instance()->core_drop(thd, dd_tspace);
dd::cache::Storage_adapter::instance()->core_store(
thd, persisted_dd_tspace.get());
// Make sure the IDs after storing are as expected.
DBUG_ASSERT(persisted_dd_schema->id() == 1);
DBUG_ASSERT(persisted_dd_tspace->id() == 1);
/*
Finally, we update the core registry of the DD tables. This must be
done in two loops to avoid issues related to overlapping ID sequences.
*/
std::vector<Table *>::const_iterator table_it = dd_tables.begin();
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end() && table_it != dd_tables.end();
++it, ++table_it) {
/*
If we are in the process of upgrading, there may not be an entry
in the dd_tables for new tables that have been added after the
version we are upgrading from.
*/
if ((*table_it) != nullptr) {
DBUG_ASSERT((*it)->entity()->name() == (*table_it)->name());
dd::cache::Storage_adapter::instance()->core_drop(thd, *table_it);
}
}
std::vector<std::unique_ptr<Table_impl>>::const_iterator persisted_it =
persisted_dd_tables.begin();
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end() &&
persisted_it != persisted_dd_tables.end();
++it, ++persisted_it) {
/*
If we are in the process of upgrading, there may not be an entry
in the persisted_dd_tables for new tables that have been added after
the version we are upgrading from.
*/
if ((*persisted_it) == nullptr) continue;
if ((*it)->property() == System_tables::Types::CORE) {
dd::cache::Storage_adapter::instance()->core_store(
thd, static_cast<Table *>((*persisted_it).get()));
}
}
}
/*
Now, the auto releaser has released the objects, and we can go ahead and
reset the shared cache.
*/
dd::cache::Shared_dictionary_cache::instance()->reset(true);
bootstrap::DD_bootstrap_ctx::instance().set_stage(bootstrap::Stage::SYNCED);
// Commit and flush tables to force re-opening using the refreshed meta data.
if (dd::end_transaction(thd, false) || dd::execute_query(thd, "FLUSH TABLES"))
return true;
// Get hold of the temporary actual and target schema names.
String_type target_schema_name;
bool target_schema_exists = false;
if (dd::tables::DD_properties::instance().get(thd, "UPGRADE_TARGET_SCHEMA",
&target_schema_name,
&target_schema_exists))
return true;
String_type actual_schema_name;
bool actual_schema_exists = false;
if (dd::tables::DD_properties::instance().get(thd, "UPGRADE_ACTUAL_SCHEMA",
&actual_schema_name,
&actual_schema_exists))
return true;
// Reset the DDSE local dictionary cache.
handlerton *ddse = ha_resolve_by_legacy_type(thd, DB_TYPE_INNODB);
if (ddse->dict_cache_reset == nullptr) return true;
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
// Skip extraneous tables during minor downgrade.
if ((*it)->entity() == nullptr) continue;
if ((*it)->property() == System_tables::Types::CORE ||
(*it)->property() == System_tables::Types::SECOND) {
ddse->dict_cache_reset(MYSQL_SCHEMA_NAME.str,
(*it)->entity()->name().c_str());
if (target_schema_exists && !target_schema_name.empty())
ddse->dict_cache_reset(target_schema_name.c_str(),
(*it)->entity()->name().c_str());
if (actual_schema_exists && !actual_schema_name.empty())
ddse->dict_cache_reset(actual_schema_name.c_str(),
(*it)->entity()->name().c_str());
}
}
/*
At this point, we're to a large extent open for business.
If there are leftover schema names from upgrade, delete them
and remove the names from the DD properties.
*/
if (target_schema_exists && !target_schema_name.empty()) {
std::stringstream ss;
ss << "DROP SCHEMA IF EXISTS " << target_schema_name;
if (dd::execute_query(thd, ss.str().c_str())) return true;
}
if (actual_schema_exists && !actual_schema_name.empty()) {
std::stringstream ss;
ss << "DROP SCHEMA IF EXISTS " << actual_schema_name;
if (dd::execute_query(thd, ss.str().c_str())) return true;
}
/*
The statements above are auto committed, so there is nothing uncommitted
at this stage. Go ahead and remove the schema keys.
*/
if (actual_schema_exists)
(void)dd::tables::DD_properties::instance().remove(thd,
"UPGRADE_ACTUAL_SCHEMA");
if (target_schema_exists)
(void)dd::tables::DD_properties::instance().remove(thd,
"UPGRADE_TARGET_SCHEMA");
if (actual_schema_exists || target_schema_exists)
return end_transaction(thd, false);
return false;
}
bool update_properties(THD *thd, const std::set<String_type> *create_set,
const std::set<String_type> *remove_set,
const String_type &target_table_schema_name) {
/*
Populate the dd properties with the SQL DDL and SE private data.
Store meta data of non-inert tables only.
*/
std::unique_ptr<dd::Properties> system_tables_props(
dd::Properties::parse_properties(""));
dd::cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
if (is_non_inert_dd_or_ddse_table((*it)->property())) {
/*
This will not be called for minor downgrade, so all tables
will have a corresponding Object_table.
*/
DBUG_ASSERT((*it)->entity() != nullptr);
const Object_table_definition *table_def =
(*it)->entity()->target_table_definition();
// May be null for abandoned tables, which should be skipped.
if (table_def == nullptr) {
continue;
}
/*
Tables that are in the remove_set, but not in the create_set,
should not be reflected in the DD properties.
*/
if (remove_set != nullptr && create_set != nullptr &&
remove_set->find((*it)->entity()->name()) != remove_set->end() &&
create_set->find((*it)->entity()->name()) == create_set->end()) {
continue;
}
/*
If a create set is submitted, use this to decide whether we should
get the meta data from the table in the 'mysql' schema or the temporary
target schema.
*/
String_type table_schema_name{MYSQL_SCHEMA_NAME.str};
if (create_set != nullptr &&
create_set->find((*it)->entity()->name()) != create_set->end()) {
table_schema_name = target_table_schema_name;
}
/*
Acquire the table object to get hold of the se private data etc.
Note that we must acquire it from the appropriate schema.
*/
const dd::Table *dd_table = nullptr;
if (thd->dd_client()->acquire(table_schema_name, (*it)->entity()->name(),
&dd_table))
return dd::end_transaction(thd, true);
// All non-abandoned tables should have a table object present.
DBUG_ASSERT(dd_table != nullptr);
std::unique_ptr<dd::Properties> tbl_props(
dd::Properties::parse_properties(""));
using dd::tables::DD_properties;
tbl_props->set(DD_properties::dd_key(DD_properties::DD_property::ID),
dd_table->se_private_id());
tbl_props->set(DD_properties::dd_key(DD_properties::DD_property::DATA),
dd_table->se_private_data().raw_string());
tbl_props->set(
DD_properties::dd_key(DD_properties::DD_property::SPACE_ID),
dd_table->tablespace_id());
// Store the structured representation of the table definition.
std::unique_ptr<Properties> definition(Properties::parse_properties(""));
table_def->store_into_properties(definition.get());
tbl_props->set(DD_properties::dd_key(DD_properties::DD_property::DEF),
definition->raw_string());
// Store the se private data for each index.
dd::Table::Index_collection::const_iterator idx(
dd_table->indexes().begin());
for (int count = 0; idx != dd_table->indexes().end(); ++idx, ++count) {
std::stringstream ss;
ss << DD_properties::dd_key(DD_properties::DD_property::IDX) << count;
tbl_props->set(ss.str().c_str(),
(*idx)->se_private_data().raw_string());
}
// Store the se private data for each column.
dd::Table::Column_collection::const_iterator col(
dd_table->columns().begin());
for (int count = 0; col != dd_table->columns().end(); ++col, ++count) {
std::stringstream ss;
ss << DD_properties::dd_key(DD_properties::DD_property::COL) << count;
tbl_props->set(ss.str().c_str(),
(*col)->se_private_data().raw_string());
}
// All tables should be reflected in the System tables list.
system_tables_props->set(dd_table->name(), tbl_props->raw_string());
}
}
if (dd::tables::DD_properties::instance().set(thd, "SYSTEM_TABLES",
*system_tables_props.get()))
return dd::end_transaction(thd, true);
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::STORED_DD_META_DATA);
// Delay commit.
return false;
}
bool update_versions(THD *thd, bool is_dd_upgrade_57) {
/*
During initialize, store the DD version number, the LCTN used, and the
mysqld server version.
*/
if (opt_initialize) {
if (dd::tables::DD_properties::instance().set(thd, "DD_VERSION",
dd::DD_VERSION) ||
dd::tables::DD_properties::instance().set(
thd, "MINOR_DOWNGRADE_THRESHOLD",
dd::DD_VERSION_MINOR_DOWNGRADE_THRESHOLD) ||
dd::tables::DD_properties::instance().set(thd, "SDI_VERSION",
dd::SDI_VERSION) ||
dd::tables::DD_properties::instance().set(thd, "LCTN",
lower_case_table_names) ||
dd::tables::DD_properties::instance().set(thd, "MYSQLD_VERSION_LO",
MYSQL_VERSION_ID) ||
dd::tables::DD_properties::instance().set(thd, "MYSQLD_VERSION_HI",
MYSQL_VERSION_ID) ||
dd::tables::DD_properties::instance().set(thd, "MYSQLD_VERSION",
MYSQL_VERSION_ID))
return dd::end_transaction(thd, true);
if (is_dd_upgrade_57) {
if (dd::tables::DD_properties::instance().set(
thd, "MYSQLD_VERSION_UPGRADED", bootstrap::SERVER_VERSION_50700))
return true;
bootstrap::DD_bootstrap_ctx::instance().set_upgraded_server_version(
bootstrap::SERVER_VERSION_50700);
} else {
if (dd::tables::DD_properties::instance().set(
thd, "MYSQLD_VERSION_UPGRADED", MYSQL_VERSION_ID))
return true;
bootstrap::DD_bootstrap_ctx::instance().set_upgraded_server_version(
MYSQL_VERSION_ID);
if (Minor_upgrade_ctx::instance()->save_and_set_extra_mvu_version(
thd,
Minor_upgrade_ctx::instance()->get_target_extra_mvu_version()))
return true;
}
} else {
uint mysqld_version_lo = 0;
uint mysqld_version_hi = 0;
uint mysqld_version = 0;
uint upgraded_server_version = 0;
bool exists_lo = false;
bool exists_hi = false;
bool exists = false;
bool exists_upgraded_version = false;
if ((dd::tables::DD_properties::instance().get(
thd, "MYSQLD_VERSION_LO", &mysqld_version_lo, &exists_lo) ||
!exists_lo) ||
(dd::tables::DD_properties::instance().get(
thd, "MYSQLD_VERSION_HI", &mysqld_version_hi, &exists_hi) ||
!exists_hi) ||
(dd::tables::DD_properties::instance().get(thd, "MYSQLD_VERSION",
&mysqld_version, &exists) ||
!exists))
return dd::end_transaction(thd, true);
if (dd::tables::DD_properties::instance().get(
thd, "MYSQLD_VERSION_UPGRADED", &upgraded_server_version,
&exists_upgraded_version) ||
!exists_upgraded_version) {
if (dd::tables::DD_properties::instance().set(
thd, "MYSQLD_VERSION_UPGRADED", mysqld_version))
return true;
upgraded_server_version = mysqld_version;
}
bootstrap::DD_bootstrap_ctx::instance().set_upgraded_server_version(
upgraded_server_version);
Minor_upgrade_ctx::instance()->retrieve_and_set_extra_mvu_version(thd);
if ((mysqld_version_lo > MYSQL_VERSION_ID &&
dd::tables::DD_properties::instance().set(thd, "MYSQLD_VERSION_LO",
MYSQL_VERSION_ID)) ||
(mysqld_version_hi < MYSQL_VERSION_ID &&
dd::tables::DD_properties::instance().set(thd, "MYSQLD_VERSION_HI",
MYSQL_VERSION_ID)) ||
(mysqld_version != MYSQL_VERSION_ID &&
dd::tables::DD_properties::instance().set(thd, "MYSQLD_VERSION",
MYSQL_VERSION_ID)))
return dd::end_transaction(thd, true);
/*
Update the SDI version number in case of upgrade.
Note that on downgrade, we keep the old SDI version.
*/
uint stored_sdi_version = 0;
bool exists_sdi = false;
if ((dd::tables::DD_properties::instance().get(
thd, "SDI_VERSION", &stored_sdi_version, &exists_sdi) ||
!exists_sdi) ||
(stored_sdi_version < dd::SDI_VERSION &&
dd::tables::DD_properties::instance().set(thd, "SDI_VERSION",
dd::SDI_VERSION)))
return dd::end_transaction(thd, true);
/*
Update the DD version number in case of upgrade.
Note that on downgrade, we keep the old DD version.
*/
uint dd_version = 0;
bool exists_dd = false;
if ((dd::tables::DD_properties::instance().get(thd, "DD_VERSION",
&dd_version, &exists_dd) ||
!exists_dd) ||
(dd_version < dd::DD_VERSION &&
dd::tables::DD_properties::instance().set(thd, "DD_VERSION",
dd::DD_VERSION)))
return dd::end_transaction(thd, true);
/*
Update the minor downgrade threshold in case of upgrade.
Note that on downgrade, we keep the threshold version which is
already present.
*/
if (dd_version < dd::DD_VERSION &&
dd::tables::DD_properties::instance().set(
thd, "MINOR_DOWNGRADE_THRESHOLD",
dd::DD_VERSION_MINOR_DOWNGRADE_THRESHOLD))
return dd::end_transaction(thd, true);
}
/*
Update the server version number in the bootstrap ctx and the
DD tablespace header if we have been doing a server upgrade.
Note that the update of the tablespace header is not rolled
back in case of an abort, so this better be the last step we
do before committing.
*/
handlerton *ddse = ha_resolve_by_legacy_type(thd, DB_TYPE_INNODB);
if (bootstrap::DD_bootstrap_ctx::instance().is_server_upgrade()) {
if (ddse->dict_set_server_version == nullptr ||
ddse->dict_set_server_version()) {
LogErr(ERROR_LEVEL, ER_CANNOT_SET_SERVER_VERSION_IN_TABLESPACE_HEADER);
return dd::end_transaction(thd, true);
}
}
#ifndef DBUG_OFF
/*
Debug code to make sure that after updating version numbers, regardless
of the type of initialization, restart or upgrade, the server version
number in the DD tablespace header is indeed the same as this server's
version number.
*/
uint version = 0;
DBUG_ASSERT(ddse->dict_get_server_version != nullptr);
DBUG_ASSERT(!ddse->dict_get_server_version(&version));
DBUG_ASSERT(version == MYSQL_VERSION_ID);
#endif
bootstrap::DD_bootstrap_ctx::instance().set_stage(
bootstrap::Stage::VERSION_UPDATED);
/*
During upgrade, this will commit the swap of the old and new DD tables.
*/
return dd::end_transaction(thd, false);
}
} // namespace dd