1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288
|
/* Copyright (c) 2017, 2025, Oracle and/or its affiliates.
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 designed to work 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 either included with
the program or referenced in the documentation.
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/upgrade_57/upgrade.h"
#include <errno.h>
#include <fcntl.h>
#include <sys/types.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#include <string>
#include <vector>
#include "lex_string.h"
#include "my_compiler.h"
#include "my_dbug.h"
#include "my_dir.h"
#include "my_inttypes.h"
#include "my_io.h"
#include "my_loglevel.h"
#include "my_sys.h"
#include "mysql/components/services/log_builtins.h"
#include "mysql/components/services/log_shared.h"
#include "mysql/plugin.h"
#include "mysql/psi/mysql_file.h" // mysql_file_open
#include "mysql/udf_registration_types.h"
#include "mysql_version.h" // MYSQL_VERSION_ID
#include "mysqld_error.h"
#include "sql/dd/cache/dictionary_client.h" // dd::cache::Dictionary_client
#include "sql/dd/dd_schema.h" // dd::schema_exists
#include "sql/dd/dd_tablespace.h" // dd::fill_table_and_parts...
#include "sql/dd/impl/bootstrap/bootstrap_ctx.h" // dd::DD_bootstrap_ctx
#include "sql/dd/impl/bootstrap/bootstrapper.h" // execute_query
#include "sql/dd/impl/dictionary_impl.h" // dd::Dictionary_impl
#include "sql/dd/impl/sdi.h" // sdi::store()
#include "sql/dd/impl/system_registry.h" // dd::System_tables
#include "sql/dd/impl/upgrade/server.h"
#include "sql/dd/impl/utils.h" // execute_query
#include "sql/dd/info_schema/metadata.h" // dd::info_schema::install_IS...
#include "sql/dd/performance_schema/init.h" // create_pfs_schema
#include "sql/dd/sdi_file.h" // dd::sdi_file::EXT
#include "sql/dd/types/object_table.h"
#include "sql/dd/types/table.h" // dd::Table
#include "sql/dd/types/tablespace.h"
#include "sql/dd/upgrade/server.h"
#include "sql/dd/upgrade_57/event.h"
#include "sql/dd/upgrade_57/global.h"
#include "sql/dd/upgrade_57/routine.h"
#include "sql/dd/upgrade_57/schema.h"
#include "sql/dd/upgrade_57/table.h"
#include "sql/error_handler.h" // Dummy_error_handler
#include "sql/handler.h"
#include "sql/lock.h" // Tablespace_hash_set
#include "sql/log.h" // sql_print_warning
#include "sql/mysqld.h" // key_file_sdi
#include "sql/sd_notify.h" // sysd::notify
#include "sql/sql_class.h" // THD
#include "sql/sql_list.h"
#include "sql/sql_plugin.h"
#include "sql/sql_plugin_ref.h"
#include "sql/sql_prepare.h"
#include "sql/sql_table.h" // build_tablename
#include "sql/stateless_allocator.h"
#include "sql/strfunc.h" // lex_cstring_handle
#include "sql/table.h"
#include "sql/thd_raii.h"
#include "sql/transaction.h" // trans_rollback
namespace dd {
namespace upgrade_57 {
/*
The variable is used to differentiate between a normal server restart
and server upgrade.
For the upgrade, before populating the DD tables, all the plugins needs
to be initialized. Once the plugins are initialized, the server calls
DD initialization function again to finish the upgrade process.
In case of deleting dictionary tables, we need to delete mysql.ibd
after innodb is shutdown. This flag is used in server shutdown code
to delete mysql.ibd after unsuccessful attempt of upgrade.
*/
static bool dd_upgrade_flag = false;
/*
This variable is used to skip creation of SDI files for tables
and InnoDB tablespaces when creating entry in dictionary. SDI entry
These entries are created as last phase of upgrade.
*/
static bool create_sdi_flag = true;
bool in_progress() { return dd_upgrade_flag; }
void set_in_progress(bool param) { dd_upgrade_flag = param; }
bool allow_sdi_creation() { return create_sdi_flag; }
void set_allow_sdi_creation(bool param) { create_sdi_flag = param; }
/**
Check if it is a file extension which should be moved
to backup_metadata_57 folder upgrade upgrade is successful.
*/
static bool check_file_extension(const String_type &extn) {
// Check for extensions
if (extn.size() < 4) return false;
return ((extn.compare(extn.size() - 4, 4, reg_ext) == 0) ||
(extn.compare(extn.size() - 4, 4, TRG_EXT) == 0) ||
(extn.compare(extn.size() - 4, 4, TRN_EXT) == 0) ||
(extn.compare(extn.size() - 4, 4, PAR_EXT) == 0) ||
(extn.compare(extn.size() - 4, 4, OPT_EXT) == 0) ||
(extn.compare(extn.size() - 4, 4, NDB_EXT) == 0) ||
(extn.compare(extn.size() - 4, 4, ISL_EXT) == 0));
}
/**
Rename stats tables. Installation from 5.7 should contain
mysql.innodb_table_stats and mysql.innodb_index_stats.
Rename .frm and .ibd files for these tables here. These tables
will be created from scratch with other dictionary tables. Data from
5.7 stats table will be inserted in new created stats table
via INSERT...SELECT statement.
*/
static void rename_stats_tables() {
/*
Rename mysql/innodb_index_stats.ibd and mysql/innodb_table_stats.ibd.
Dictionary bootstrap will create these tables. Upgrade will copy
data from 5.7 version of these tables using INSERT..SELECT
*/
char to_path[FN_REFLEN];
char from_path[FN_REFLEN];
bool not_used;
build_table_filename(to_path, sizeof(to_path) - 1, MYSQL_SCHEMA_NAME.str,
index_stats_backup.c_str(), IBD_EXT.c_str(), 0,
¬_used);
build_table_filename(from_path, sizeof(from_path) - 1, MYSQL_SCHEMA_NAME.str,
index_stats.c_str(), IBD_EXT.c_str(), 0, ¬_used);
if (mysql_file_rename(key_file_misc, from_path, to_path, MYF(0))) {
LogErr(WARNING_LEVEL, ER_DD_UPGRADE_RENAME_IDX_STATS_FILE_FAILED);
}
build_table_filename(to_path, sizeof(to_path) - 1, MYSQL_SCHEMA_NAME.str,
table_stats_backup.c_str(), IBD_EXT.c_str(), 0,
¬_used);
build_table_filename(from_path, sizeof(from_path) - 1, MYSQL_SCHEMA_NAME.str,
table_stats.c_str(), IBD_EXT.c_str(), 0, ¬_used);
if (mysql_file_rename(key_file_misc, from_path, to_path, MYF(0))) {
LogErr(WARNING_LEVEL, ER_DD_UPGRADE_RENAME_IDX_STATS_FILE_FAILED);
}
}
/**
Cleanup inside SE after upgrade for one SE.
@param[in] thd Thread Handle
@param[in] plugin Handlerton Plugin
@param[in] failed_upgrade Flag to tell SE if cleanup is after failed
failed upgrade or successful upgrade.
@retval false ON SUCCESS
@retval true ON FAILURE
*/
static bool ha_finish_upgrade(THD *thd, plugin_ref plugin,
void *failed_upgrade) {
handlerton *hton = plugin_data<handlerton *>(plugin);
if (hton->finish_upgrade) {
if (hton->finish_upgrade(thd, *(static_cast<bool *>(failed_upgrade))))
return true;
}
return false;
}
// Cleanup inside SE after upgrade for all SE.
bool ha_finish_upgrade(THD *thd, bool failed_upgrade) {
return (plugin_foreach(thd, ha_finish_upgrade, MYSQL_STORAGE_ENGINE_PLUGIN,
static_cast<void *>(&failed_upgrade)));
}
/**
In case of successful upgrade, this function
deletes all .frm, .TRG, .TRG, .par, .opt, .isl
files from all databases.
mysql.proc, mysql.event, statistics tables from 5.7
will be deleted as part of cleanup.
@param[in] thd Thread handle.
*/
bool finalize_upgrade(THD *thd) {
uint i;
MY_DIR *a = nullptr, *b = nullptr;
String_type path;
char from_path[FN_REFLEN];
std::vector<String_type> db_name;
Upgrade_status().remove();
// Drop mysql.proc and mysql.event tables from 5.7
(void)dd::execute_query(thd, "DROP TABLE IF EXISTS mysql.proc");
(void)dd::execute_query(thd, "DROP TABLE IF EXISTS mysql.event");
(void)dd::execute_query(thd,
"DROP TABLE IF EXISTS "
"mysql.innodb_table_stats_backup57");
(void)dd::execute_query(thd,
"DROP TABLE IF EXISTS "
"mysql.innodb_index_stats_backup57");
path.assign(mysql_real_data_home);
if (!(a = my_dir(path.c_str(), MYF(MY_WANT_STAT)))) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_DD_OPEN_FAILED, path.c_str());
return true;
}
// Scan all files and folders in data directory.
for (i = 0; i < (uint)a->number_off_files; i++) {
String_type file;
file.assign(a->dir_entry[i].name);
if (file.at(0) == '.') continue;
// If its a folder, add it to the vector.
if (MY_S_ISDIR(a->dir_entry[i].mystat->st_mode)) {
db_name.push_back(a->dir_entry[i].name);
} else {
String_type file_ext;
if (file.size() < 4) continue;
file_ext.assign(file.c_str() + file.size() - 4);
// Get the name without the file extension.
if (check_file_extension(file_ext)) {
if (fn_format(from_path, file.c_str(), mysql_real_data_home, "",
MYF(MY_UNPACK_FILENAME | MY_SAFE_PATH)) == nullptr)
return true;
(void)mysql_file_delete(key_file_misc, from_path, MYF(0));
}
}
}
// Iterate through the databases list
for (String_type str : db_name) {
String_type dir_name = str.c_str();
char dir_path[FN_REFLEN];
if (fn_format(dir_path, dir_name.c_str(), path.c_str(), "",
MYF(MY_UNPACK_FILENAME | MY_SAFE_PATH)) == nullptr)
continue;
if (!(b = my_dir(dir_path, MYF(MY_WANT_STAT)))) continue;
// Scan all files and folders in data directory.
for (i = 0; i < (uint)b->number_off_files; i++) {
String_type file;
file.assign(b->dir_entry[i].name);
if ((file.at(0) == '.') || (file.size() < 4)) continue;
String_type file_ext;
file_ext.assign(file.c_str() + file.size() - 4);
// Get the name without the file extension.
if (check_file_extension(file_ext)) {
if (fn_format(from_path, file.c_str(), dir_path, "",
MYF(MY_UNPACK_FILENAME | MY_SAFE_PATH)) == nullptr)
continue;
(void)mysql_file_delete(key_file_misc, from_path, MYF(0));
}
}
my_dirend(b);
}
my_dirend(a);
return ha_finish_upgrade(thd, false);
}
/**
Function to scan mysql schema to check if any tables exist
with the same name as DD tables to be created.
This function checks existence of .frm files in mysql schema.
@retval false ON SUCCESS
@retval true ON FAILURE
*/
bool check_for_dd_tables() {
// Iterate over DD tables, check .frm files
for (System_tables::Const_iterator it = System_tables::instance()->begin();
it != System_tables::instance()->end(); ++it) {
if ((*it)->property() == System_tables::Types::SYSTEM) {
continue;
}
String_type table_name = (*it)->entity()->name();
String_type schema_name(MYSQL_SCHEMA_NAME.str);
const System_tables::Types *table_type =
System_tables::instance()->find_type(schema_name, table_name);
bool is_innodb_stats_table =
(table_type != nullptr) &&
(*table_type == System_tables::Types::DDSE_PROTECTED);
is_innodb_stats_table &= ((table_name == "innodb_table_stats") ||
(table_name == "innodb_index_stats"));
if (is_innodb_stats_table) continue;
char path[FN_REFLEN + 1];
bool not_used;
build_table_filename(path, sizeof(path) - 1, "mysql", table_name.c_str(),
reg_ext, 0, ¬_used);
if (!my_access(path, F_OK)) {
LogErr(ERROR_LEVEL, ER_FILE_EXISTS_DURING_UPGRADE, path);
return true;
}
}
return false;
}
/**
Rename back .ibd files for innodb stats table.
*/
void rename_back_stats_tables(THD *thd) {
/*
Ignore error of the statement execution as we are already in
error handling code.
*/
(void)dd::execute_query(thd, "SET GLOBAL INNODB_FAST_SHUTDOWN= 0");
// Rename back mysql/innodb_index_stats.ibd and mysql/innodb_table_stats.ibd
char to_path[FN_REFLEN];
char from_path[FN_REFLEN];
bool not_used;
build_table_filename(to_path, sizeof(to_path) - 1, MYSQL_SCHEMA_NAME.str,
index_stats.c_str(), IBD_EXT.c_str(), 0, ¬_used);
build_table_filename(from_path, sizeof(from_path) - 1, MYSQL_SCHEMA_NAME.str,
index_stats_backup.c_str(), IBD_EXT.c_str(), 0,
¬_used);
(void)mysql_file_rename(key_file_misc, from_path, to_path, MYF(0));
build_table_filename(to_path, sizeof(to_path) - 1, MYSQL_SCHEMA_NAME.str,
table_stats.c_str(), IBD_EXT.c_str(), 0, ¬_used);
build_table_filename(from_path, sizeof(from_path) - 1, MYSQL_SCHEMA_NAME.str,
table_stats_backup.c_str(), IBD_EXT.c_str(), 0,
¬_used);
(void)mysql_file_rename(key_file_misc, from_path, to_path, MYF(0));
}
/**
Drop all .SDI files created during upgrade.
*/
static void drop_sdi_files() {
uint i, j;
// Iterate in data directory and delete all .SDI files
MY_DIR *a, *b;
String_type path;
path.assign(mysql_real_data_home);
if (!(a = my_dir(path.c_str(), MYF(MY_WANT_STAT)))) {
LogErr(ERROR_LEVEL, ER_CANT_OPEN_DATADIR_AFTER_UPGRADE_FAILURE,
path.c_str());
return;
}
// Scan all files and folders in data directory.
for (i = 0; i < (uint)a->number_off_files; i++) {
String_type file;
file.assign(a->dir_entry[i].name);
if (file.at(0) == '.') continue;
// If its a folder, iterate it to delete all .SDI files
if (MY_S_ISDIR(a->dir_entry[i].mystat->st_mode)) {
char dir_path[FN_REFLEN];
if (fn_format(dir_path, file.c_str(), path.c_str(), "",
MYF(MY_UNPACK_FILENAME | MY_SAFE_PATH)) == nullptr) {
LogErr(ERROR_LEVEL, ER_CANT_SET_PATH_FOR, file.c_str());
continue;
}
if (!(b = my_dir(dir_path, MYF(MY_WANT_STAT)))) {
LogErr(ERROR_LEVEL, ER_CANT_OPEN_DIR, dir_path);
continue;
}
// Scan all files and folders in data directory.
for (j = 0; j < (uint)b->number_off_files; j++) {
String_type file2;
file2.assign(b->dir_entry[j].name);
if ((file2.at(0) == '.') || (file2.size() < 4)) continue;
String_type file_ext;
file_ext.assign(file2.c_str() + file2.size() - 4);
if (file_ext.compare(0, 4, dd::sdi_file::EXT) == 0) {
char to_path[FN_REFLEN];
if (fn_format(to_path, file2.c_str(), dir_path, "",
MYF(MY_UNPACK_FILENAME | MY_SAFE_PATH)) == nullptr) {
LogErr(ERROR_LEVEL, ER_CANT_SET_PATH_FOR, file2.c_str());
continue;
}
(void)mysql_file_delete(key_file_sdi, to_path, MYF(MY_WME));
}
}
my_dirend(b);
} else {
// Delete .SDI files in data directory created for schema.
String_type file_ext;
if (file.size() < 4) continue;
file_ext.assign(file.c_str() + file.size() - 4);
// Get the name without the file extension.
if (file_ext.compare(0, 4, dd::sdi_file::EXT) == 0) {
char to_path[FN_REFLEN];
if (fn_format(to_path, file.c_str(), path.c_str(), "",
MYF(MY_UNPACK_FILENAME | MY_SAFE_PATH)) == nullptr) {
LogErr(ERROR_LEVEL, ER_CANT_SET_PATH_FOR, file.c_str());
continue;
}
(void)mysql_file_delete(key_file_sdi, to_path, MYF(MY_WME));
}
}
}
my_dirend(a);
} // drop_dd_table
/**
Create SDI information for all tablespaces and tables.
@param[in] thd Thread handle.
@retval false ON SUCCESS
@retval true ON FAILURE
*/
bool add_sdi_info(THD *thd) {
// Fetch list of tablespaces. We will ignore error in storing SDI info
// as upgrade can only roll forward in this stage. Use Error handler to avoid
// any error calls in dd::sdi::store()
std::vector<dd::String_type> tablespace_names;
Dummy_error_handler error_handler;
MEM_ROOT mem_root(PSI_NOT_INSTRUMENTED, MEM_ROOT_BLOCK_SIZE);
Thd_mem_root_guard root_guard(thd, &mem_root);
if (thd->dd_client()->fetch_global_component_names<dd::Tablespace>(
&tablespace_names)) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FAILED_TO_FETCH_TABLESPACES);
return (true);
}
// Add sdi info
thd->push_internal_handler(&error_handler);
for (dd::String_type &tsc : tablespace_names) {
dd::cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
Disable_autocommit_guard autocommit_guard(thd);
dd::Tablespace *ts = nullptr;
if (thd->dd_client()->acquire_for_modification<dd::Tablespace>(tsc, &ts)) {
// In case of error, we will continue with upgrade.
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FAILED_TO_ACQUIRE_TABLESPACE,
tsc.c_str());
continue;
}
plugin_ref pr =
ha_resolve_by_name_raw(thd, lex_cstring_handle(ts->engine()));
handlerton *hton = nullptr;
if (pr)
hton = plugin_data<handlerton *>(pr);
else
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FAILED_TO_RESOLVE_TABLESPACE_ENGINE,
ts->name().c_str(), ts->engine().c_str());
// In case of error, we will continue with upgrade.
if (hton && hton->upgrade_space_version(ts))
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FAILED_TO_UPDATE_VER_NO_IN_TABLESPACE,
ts->name().c_str());
if (hton && hton->sdi_create) {
// Error handling not possible at this stage, upgrade should complete.
if (hton->sdi_create(ts))
LogErr(ERROR_LEVEL, ER_FAILED_TO_STORE_SDI_FOR_TABLESPACE,
ts->name().c_str());
// Write changes to dictionary.
if (thd->dd_client()->update(ts)) {
trans_rollback_stmt(thd);
LogErr(ERROR_LEVEL, ER_FAILED_TO_STORE_SDI_FOR_TABLESPACE,
ts->name().c_str());
}
trans_commit_stmt(thd);
trans_commit(thd);
}
mem_root.ClearForReuse();
}
thd->pop_internal_handler();
// Fetch list of tables from dictionary
std::vector<dd::Object_id> table_ids;
if (thd->dd_client()->fetch_global_component_ids<dd::Table>(&table_ids)) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FAILED_TO_FETCH_TABLES);
return (true);
}
// Add sdi info
thd->push_internal_handler(&error_handler);
for (dd::Object_id &table_id : table_ids) {
dd::cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
const dd::Table *table = nullptr;
if (thd->dd_client()->acquire(table_id, &table) || !table) {
mem_root.ClearForReuse();
continue;
}
if (dd::sdi::store(thd, table)) {
LogErr(ERROR_LEVEL, ER_UNKNOWN_TABLE_IN_UPGRADE, table->name().c_str());
trans_rollback_stmt(thd);
}
trans_commit_stmt(thd);
trans_commit(thd);
mem_root.ClearForReuse();
}
thd->pop_internal_handler();
// Add status to mark tablespace modification complete.
if (Upgrade_status().update(Upgrade_status::enum_stage::SDI_INFO_UPDATED))
return true;
LogErr(SYSTEM_LEVEL, ER_DD_UPGRADE_DD_POPULATED);
sysd::notify("STATUS=Data Dictionary upgrade from MySQL 5.7 complete\n");
return false;
} // add_sdi_info
//
// Methods of class Update_status.
//
// Get update status.
Upgrade_status::enum_stage Upgrade_status::get() {
if (open(O_RDONLY)) return enum_stage::NONE;
enum_stage status = read();
if (close()) return enum_stage::NONE;
return status;
}
// Update upgrade status.
bool Upgrade_status::update(Upgrade_status::enum_stage stage) {
if (open(O_TRUNC | O_WRONLY)) return true;
if (write(stage)) return true;
if (close()) return true;
return false;
}
// Constructor initialization.
Upgrade_status::Upgrade_status()
: m_file(nullptr), m_filename("mysql_dd_upgrade_info") {}
// Create status file.
bool Upgrade_status::create() {
if (open(O_TRUNC | O_WRONLY)) return true;
if (write(enum_stage::STARTED)) return true;
if (exists() == false || close()) return true;
return false;
}
// Open status file.
bool Upgrade_status::open(int flags) {
assert(m_file == nullptr);
if (!(m_file = my_fopen(m_filename.c_str(), flags, MYF(0)))) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_INFO_FILE_OPEN_FAILED, m_filename.c_str(),
errno);
return true;
}
return false;
}
// Read status from file.
Upgrade_status::enum_stage Upgrade_status::read() {
assert(m_file);
enum_stage stage = enum_stage::NONE;
size_t items_read [[maybe_unused]];
if (!feof(m_file)) items_read = fread(&stage, sizeof(int), 1, m_file);
return stage;
}
// Write status to file.
bool Upgrade_status::write(Upgrade_status::enum_stage stage) {
assert(m_file);
if (fwrite(&stage, sizeof(int), 1, m_file) != 1) {
char errbuf[MYSYS_STRERROR_SIZE];
LogErr(ERROR_LEVEL, ER_FAILED_TO_WRITE_TO_FILE, m_filename.c_str(),
ferror(m_file), my_strerror(errbuf, sizeof(errbuf), ferror(m_file)));
return true;
}
fflush(m_file);
return false;
}
bool Upgrade_status::exists() {
// Check if the upgrade_info_file was properly created/updated
return !my_access(m_filename.c_str(), F_OK);
}
// Close status file.
bool Upgrade_status::close() {
assert(m_file);
if (my_fclose(m_file, MYF(0))) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_INFO_FILE_CLOSE_FAILED,
m_filename.c_str(), errno);
return true;
}
m_file = nullptr;
return false;
}
// Delete status file.
bool Upgrade_status::remove() {
assert(!m_file);
(void)mysql_file_delete(key_file_misc, m_filename.c_str(), MYF(MY_WME));
return false;
}
// Delete dictionary tables
bool terminate(THD *thd) {
// Function call to SEs for cleanup after failed upgrade.
ha_finish_upgrade(thd, true);
// RAII to handle error messages.
dd::upgrade::Bootstrap_error_handler bootstrap_error_handler;
// Set flag true to delete mysql.ibd after innodb is shutdown.
set_in_progress(true);
// Rename back stats tables
rename_back_stats_tables(thd);
// Drop SDI files.
drop_sdi_files();
return false;
}
/**
create data dictionary entry for tablespaces for one SE.
@param[in] thd Thread Handle
@param[in] plugin Handlerton Plugin
@retval false ON SUCCESS
@retval true ON FAILURE
*/
static bool ha_migrate_tablespaces(THD *thd, plugin_ref plugin, void *) {
handlerton *hton = plugin_data<handlerton *>(plugin);
int error = 0;
if (hton->upgrade_tablespace) {
error = hton->upgrade_tablespace(thd);
// Commit or Rollback dictionary change.
if (error) {
trans_rollback_stmt(thd);
// Full rollback in case we have THD::transaction_rollback_request.
trans_rollback(thd);
} else {
error = (trans_commit_stmt(thd) || trans_commit(thd));
}
if (error) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_TABLESPACE_MIGRATION_FAILED, error);
return true;
}
}
return false;
}
/**
create data dictionary entry for tablespaces for one SE.
@param[in] thd Thread Handle
@retval false ON SUCCESS
@retval true ON FAILURE
*/
static bool ha_migrate_tablespaces(THD *thd) {
return (plugin_foreach(thd, ha_migrate_tablespaces,
MYSQL_STORAGE_ENGINE_PLUGIN, nullptr));
}
/**
Migrate statistics from 5.7 stats tables.
Ignore error here as stats table data can be regenerated by
user using ANALYZE command.
*/
static bool migrate_stats(THD *thd) {
dd::upgrade::Bootstrap_error_handler error_handler;
error_handler.set_log_error(false);
if (dd::execute_query(thd,
"INSERT IGNORE INTO mysql.innodb_table_stats "
"SELECT * FROM mysql.innodb_table_stats_backup57 "
"WHERE table_name not like '%#P#%'"))
LogErr(WARNING_LEVEL, ER_DD_UPGRADE_FAILED_TO_CREATE_TABLE_STATS);
else
LogErr(INFORMATION_LEVEL, ER_DD_UPGRADE_TABLE_STATS_MIGRATE_COMPLETED);
if (dd::execute_query(thd,
"INSERT IGNORE INTO mysql.innodb_index_stats "
"SELECT * FROM mysql.innodb_index_stats_backup57 "
"WHERE table_name not like '%#P#%'"))
LogErr(WARNING_LEVEL, ER_DD_UPGRADE_FAILED_TO_CREATE_INDEX_STATS);
else
LogErr(INFORMATION_LEVEL, ER_DD_UPGRADE_TABLE_STATS_MIGRATE_COMPLETED);
// Reset error logging
error_handler.set_log_error(true);
return false;
}
// Initialize dictionary in case of server restart.
static bool restart_dictionary(THD *thd) {
// RAII to handle error messages.
dd::upgrade::Bootstrap_error_handler bootstrap_error_handler;
// 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.
*/
bool error = false;
thd->push_internal_handler(&key_error_handler);
bootstrap_error_handler.set_log_error(false);
error = bootstrap::restart(thd);
bootstrap_error_handler.set_log_error(true);
thd->pop_internal_handler();
return error;
}
/**
Upgrade logs inside one SE.
@param[in] thd Thread Handle
@param[in] plugin Handlerton Plugin
@retval false ON SUCCESS
@retval true ON FAILURE
*/
static bool upgrade_logs(THD *thd, plugin_ref plugin, void *) {
handlerton *hton = plugin_data<handlerton *>(plugin);
if (hton->upgrade_logs) {
if (hton->upgrade_logs(thd)) return true;
}
return false;
}
/**
Upgrade logs inside all SE.
@param[in] thd Thread Handle
@retval false ON SUCCESS
@retval true ON FAILURE
*/
static bool ha_upgrade_engine_logs(THD *thd) {
if (plugin_foreach(thd, upgrade_logs, MYSQL_STORAGE_ENGINE_PLUGIN, nullptr))
return true;
return false;
}
// Initialize DD in case of upgrade.
bool do_pre_checks_and_initialize_dd(THD *thd) {
// Set both variables false in the beginning
set_in_progress(false);
opt_initialize = false;
set_allow_sdi_creation(true);
Disable_autocommit_guard autocommit_guard(thd);
Dictionary_impl *d = dd::Dictionary_impl::instance();
assert(d);
cache::Dictionary_client::Auto_releaser releaser(thd->dd_client());
char path[FN_REFLEN + 1];
bool not_used;
build_table_filename(path, sizeof(path) - 1, "", "mysql", ".ibd", 0,
¬_used);
bool exists_mysql_tablespace = (!my_access(path, F_OK));
// Check existence of mysql/plugin.frm
build_table_filename(path, sizeof(path) - 1, "mysql", "plugin", ".frm", 0,
¬_used);
bool exists_plugin_frm = (!my_access(path, F_OK));
/*
If mysql.ibd and mysql/plugin.frm do not exist,
this is neither a restart nor an in-place upgrade case.
Upgrade process has dependency on mysql.plugin table.
Server restart is not possible without mysql.ibd.
Exit with an error.
*/
if (!exists_mysql_tablespace && !exists_plugin_frm) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FAILED_FIND_VALID_DATA_DIR);
return true;
}
// Read stage of upgrade from the file.
Upgrade_status upgrade_status;
bool upgrade_status_exists = upgrade_status.exists();
Upgrade_status::enum_stage upgrade_stage = Upgrade_status::enum_stage::NONE;
if (upgrade_status_exists) {
upgrade_stage = upgrade_status.get();
DBUG_EXECUTE_IF("dd_upgrade_debug_info",
sql_print_information("Status of upgrade is %d",
static_cast<int>(upgrade_stage)););
}
// Upgrade data directory from mysql-5.7
if (!exists_mysql_tablespace && !upgrade_status_exists) {
if (opt_upgrade_mode == UPGRADE_NONE) {
LogErr(ERROR_LEVEL, ER_SERVER_UPGRADE_OFF);
return true;
}
// Safeguard against upgrading from 5.7 to 9.x
if (MYSQL_VERSION_ID >= 90000) {
LogErr(ERROR_LEVEL, ER_INVALID_SERVER_UPGRADE_NOT_LTS,
bootstrap::SERVER_VERSION_50700, MYSQL_VERSION_ID,
bootstrap::SERVER_VERSION_50700);
return true;
}
// Create the file to track stages of upgrade.
if (upgrade_status.create()) return true;
/*
If mysql.idb does not exist and upgrade stage tracking file
does not exist, we are in upgrade mode.
*/
LogErr(SYSTEM_LEVEL, ER_DD_UPGRADE_START);
sysd::notify("STATUS=Data Dictionary upgrade from MySQL 5.7 in progress\n");
}
/*
Initialize InnoDB in restart mode if mysql.ibd is present.
Else, initialize InnoDB in upgrade mode to create mysql tablespace
and upgrade redo and undo logs.
If mysql.ibd does not exist but upgrade stage tracking file exist
This can happen in rare scenario when server detects it needs to upgrade.
Server creates mysql_dd_upgrade_info file but crashes/killed before
creating mysql.ibd. In this case, innodb was initialized above in upgrade
mode. It would create mysql tablespace. Do nothing here, we will treat this
as upgrade.
*/
if (exists_mysql_tablespace) {
if (bootstrap::DDSE_dict_init(thd, DICT_INIT_CHECK_FILES,
d->get_target_dd_version())) {
LogErr(ERROR_LEVEL, ER_DD_SE_INIT_FAILED);
return true;
}
} else {
if (bootstrap::DDSE_dict_init(thd, DICT_INIT_UPGRADE_57_FILES,
d->get_target_dd_version())) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FAILED_INIT_DD_SE);
Upgrade_status().remove();
return true;
}
}
/*
Add status to mark initialization of InnoDB.
This indicates undo and redo logs are upgraded and mysql.ibd exists.
*/
if (!exists_mysql_tablespace && !upgrade_status_exists) {
if (upgrade_status.update(Upgrade_status::enum_stage::DICT_SPACE_CREATED))
return true;
DBUG_EXECUTE_IF("dd_upgrade_stage_1",
/*
Server will crash will upgrading 5.7 data directory.
This will leave server is an inconsistent state.
File tracking upgrade will have Stage 1 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 mysql.ibd exists and upgrade stage tracking does not exist, restart
the server.
For ordinary restart of an 8.0 server, and for upgrades post 8.0,
this code path will be taken.
*/
if (exists_mysql_tablespace && !upgrade_status_exists) {
return (restart_dictionary(thd));
}
if (exists_mysql_tablespace && upgrade_status_exists) {
/*
If mysql.idb exist and upgrade stage tracking file exist,
get stage of upgrade. This is happen only in extreme cases like
Server crash or server kill.
*/
switch (upgrade_stage) {
case Upgrade_status::enum_stage::STARTED:
case Upgrade_status::enum_stage::DICT_SPACE_CREATED: {
/*
Upgrade Stage 0 DD_UPGRADE_STARTED:
File tracking upgrade stage was created but InnoDB was not completely
initialized but mysql.ibd was created. No new undo logs by InnoDB is
created.
Upgrade Stage 1 DD_UPGRADE_DICT_SPACE_CREATED:
Dictionary tables were not created completely, InnoDB undo log should
not have any new data.
Error out, delete mysql.ibd, downgrade innodb undo and redo logs.
*/
LogErr(ERROR_LEVEL, ER_DD_ABORTING_PARTIAL_UPGRADE);
terminate(thd);
return true;
}
case Upgrade_status::enum_stage::DICT_TABLES_CREATED:
case Upgrade_status::enum_stage::DICTIONARY_CREATED: {
/*
Upgrade Stage 2 DD_UPGRADE_DICT_TABLES_CREATED:
It indicates that dictionary tables were created but dictionary
was not completely initialized. This is a rare condition.
Upgrade Stage 3 DD_UPGRADE_DICTIONARY_CREATED:
It indicates that dictionary tables inititlization was complete but
user tables were not completely upgraded.
InnoDB will have undo logs in these both stages.
Inititialize dictionary, start InnoDB recovery to empty undo logs,
then error out and delete mysql.ibd, downgrade innodb undo
and redo logs.
*/
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_FOUND_PARTIALLY_UPGRADED_DD_ABORT);
// Try to Initialize dictionary to empty undo log.
bootstrap::recover_innodb_upon_upgrade(thd);
terminate(thd);
return true;
}
case Upgrade_status::enum_stage::USER_TABLE_UPGRADED: {
/*
It indicates that user tables were upgraded but SDI information was
not updated in tablespaces.
Restart dictionary, then update SDI information.
*/
LogErr(INFORMATION_LEVEL,
ER_DD_UPGRADE_FOUND_PARTIALLY_UPGRADED_DD_CONTINUE);
if (restart_dictionary(thd)) return true;
// Ignore error in this stage and continue with server restart.
(void)add_sdi_info(thd);
(void)migrate_stats(thd);
// Cleanup after successful upgrade.
finalize_upgrade(thd);
return false;
}
case Upgrade_status::enum_stage::SDI_INFO_UPDATED: {
/*
It indicates that SDI information was created but stats migration
was not complete. Ignore and continue with server restart.
*/
LogErr(INFORMATION_LEVEL,
ER_DD_UPGRADE_FOUND_PARTIALLY_UPGRADED_DD_CONTINUE);
if (restart_dictionary(thd)) return true;
// Cleanup after successful upgrade.
finalize_upgrade(thd);
return false;
}
case Upgrade_status::enum_stage::NONE:
// Try to restart server in case this impossible scenario hits
return restart_dictionary(thd);
} // End of switch
}
/*
Create New DD tables in DD and storage engine. Mark
dd_upgrade_flag to true to indicate that we are upgrading.
*/
set_allow_sdi_creation(true);
set_in_progress(true);
opt_initialize = true;
if (check_for_dd_tables()) {
LogErr(ERROR_LEVEL, ER_DD_FRM_EXISTS_FOR_TABLE);
return true;
}
/*
Ignore ER_TOO_LONG_KEY for dictionary tables creation.
TODO: Workaround due to bug#20629014. Remove when the bug is fixed.
*/
// RAII to handle error in execution of CREATE TABLE.
Key_length_error_handler key_error_handler;
thd->push_internal_handler(&key_error_handler);
if (bootstrap::initialize_dictionary(thd, in_progress(), d) ||
dd::info_schema::create_system_views(thd) ||
dd::info_schema::store_server_I_S_metadata(thd)) {
thd->pop_internal_handler();
terminate(thd);
return true;
}
// Add status to mark creation and initialization of dictionary.
if (Upgrade_status().update(Upgrade_status::enum_stage::DICTIONARY_CREATED))
return true;
thd->pop_internal_handler();
LogErr(INFORMATION_LEVEL, ER_DD_CREATED_FOR_UPGRADE);
// Rename .ibd files for innodb stats tables
rename_stats_tables();
// Mark opt_initiazlize false after creating dictionary tables.
opt_initialize = false;
// Mark flag true to skip creation of SDI information in tablespaces.
set_allow_sdi_creation(false);
// Migrate tablespaces from SE to dictionary.
if (ha_migrate_tablespaces(thd)) {
terminate(thd);
return true;
}
/*
Migrate meta data of plugin table to DD.
It is used in plugin initialization.
*/
if (migrate_plugin_table_to_dd(thd)) {
terminate(thd);
return true;
}
/*
Plugins may need to create performance schema tables. During upgrade from
5.7, we do not yet have an entry in mysql.schemata for performance schema,
so creation of such tables will fail. To avoid this, we migrate the entry
here if the schema was present in 5.7. If the performance schema was not
present in 5.7, then we create the schema explicitly, if the server is
configured to use the performance schema.
*/
size_t path_len = build_table_filename(
path, sizeof(path) - 1, PERFORMANCE_SCHEMA_DB_NAME.str, "", "", 0);
path[path_len - 1] = 0; // Remove last '/' from path
MY_STAT stat_info;
// RAII to handle error messages.
dd::upgrade::Bootstrap_error_handler bootstrap_error_handler;
if (mysql_file_stat(key_file_misc, path, &stat_info, MYF(0)) != nullptr) {
if (migrate_schema_to_dd(thd, PERFORMANCE_SCHEMA_DB_NAME.str)) {
terminate(thd);
return true;
}
}
#ifdef WITH_PERFSCHEMA_STORAGE_ENGINE
else if (dd::performance_schema::create_pfs_schema(thd)) {
terminate(thd);
return true;
}
#endif
// Reset flag
set_allow_sdi_creation(true);
return false;
}
// Server Upgrade from 5.7
bool fill_dd_and_finalize(THD *thd) {
bool error = false;
// RAII to handle error messages.
dd::upgrade::Bootstrap_error_handler bootstrap_error_handler;
/*
While migrating tables, mysql_prepare_create_table() is called which checks
for duplicated value in SET data type. Error is reported for duplicated
values only in strict sql mode. Reset the value of sql_mode to zero while
migrating data to dictionary.
*/
thd->variables.sql_mode = 0;
std::vector<dd::String_type> db_name;
std::vector<dd::String_type>::iterator it;
if (find_schema_from_datadir(&db_name)) {
terminate(thd);
return true;
}
dd::upgrade::Syntax_error_handler error_handler;
thd->push_internal_handler(&error_handler);
// Upgrade schema and tables, create view without resolving dependency
for (it = db_name.begin(); it != db_name.end(); it++) {
bool exists = false;
dd::schema_exists(thd, it->c_str(), &exists);
if (!exists && migrate_schema_to_dd(thd, it->c_str())) {
thd->pop_internal_handler();
terminate(thd);
return true;
}
// Mark flag false to skip creation of SDI information.
set_allow_sdi_creation(false);
if (migrate_all_frm_to_dd(thd, it->c_str(), false)) {
// Don't return from here, we want to print all error to error log
error |= true;
}
// Reset flag
set_allow_sdi_creation(true);
}
/*
Do not print error while resolving routine or view dependency from
my_error(). Function resolving routine/view dependency will print warning
if it is not from sys schema. Fatal errors will result in termination
of upgrade.
*/
bootstrap_error_handler.set_log_error(false);
std::set<uint> allowed_errors = {ER_DEFINITION_CONTAINS_INVALID_STRING};
bootstrap_error_handler.set_allowlist_errors(allowed_errors);
error |= migrate_events_to_dd(thd);
error |= migrate_routines_to_dd(thd);
bootstrap_error_handler.clear_allowlist_errors();
// We will not get error in this step unless its a fatal error.
for (it = db_name.begin(); it != db_name.end(); it++) {
// Upgrade view resolving dependency
if (migrate_all_frm_to_dd(thd, it->c_str(), true)) {
// Don't return from here, we want to print all error to error log.
error = true;
}
}
// Reset error log output behavior.
bootstrap_error_handler.set_log_error(true);
thd->pop_internal_handler();
DBUG_EXECUTE_IF("dd_upgrade_stage_3",
/*
Server will crash will upgrading 5.7 data directory.
This will leave server is an inconsistent state.
File tracking upgrade will have Stage 3 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 (error) {
terminate(thd);
return true;
}
// Upgrade logs in storage engine
if (ha_upgrade_engine_logs(thd)) {
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_SE_LOGS_FAILED);
terminate(thd);
return true;
}
/*
Add status to mark creation and initialization of dictionary.
We will be modifying innodb tablespaces now.
After this step, upgrade process can only roll forward.
*/
if (Upgrade_status().update(Upgrade_status::enum_stage::USER_TABLE_UPGRADED))
return true;
// Add SDI information to all tablespaces
if (add_sdi_info(thd))
LogErr(ERROR_LEVEL, ER_DD_UPGRADE_SDI_INFO_UPDATE_FAILED);
// Migrate Statistics tables
(void)migrate_stats(thd);
// Finalize upgrade process.
if (finalize_upgrade(thd)) {
terminate(thd);
return true;
}
// Mark upgrade flag false
set_in_progress(false);
if (bootstrap::setup_dd_objects_and_collations(thd)) {
terminate(thd);
return true;
}
return false;
}
} // namespace upgrade_57
} // namespace dd
|