summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorVladislav Vaintroub <wlad@mariadb.com>2017-04-18 19:05:57 +0000
committerSergei Golubchik <serg@mariadb.org>2017-04-27 19:12:40 +0200
commitce4c56db0c5806ccbea8ad78585f95f6c335b0ea (patch)
tree06c75c921b6947320e86487fc7a41dbb55307d95
parentd7714308e014a7335e1c0f60dc0afbf5eba5ff99 (diff)
downloadmariadb-git-ce4c56db0c5806ccbea8ad78585f95f6c335b0ea.tar.gz
MDEV-9566 Port Percona Xtrabackup to MariaDB as mariabackup
- Modify backup code to work with MariaDB's 10.1 xtradb - Remove crypt encryption, version_check.pl, "compact backup" - Add encryption plugin
-rw-r--r--extra/mariabackup/CMakeLists.txt235
-rw-r--r--extra/mariabackup/backup_copy.cc141
-rw-r--r--extra/mariabackup/backup_copy.h2
-rw-r--r--extra/mariabackup/backup_mysql.cc262
-rw-r--r--extra/mariabackup/changed_page_bitmap.cc2
-rw-r--r--extra/mariabackup/common.h48
-rw-r--r--extra/mariabackup/compact.cc1059
-rw-r--r--extra/mariabackup/compact.h44
-rw-r--r--extra/mariabackup/datasink.c9
-rw-r--r--extra/mariabackup/datasink.h1
-rw-r--r--extra/mariabackup/ds_compress.c4
-rw-r--r--extra/mariabackup/ds_encrypt.c3
-rw-r--r--extra/mariabackup/ds_encrypt.h4
-rw-r--r--extra/mariabackup/ds_local.c8
-rw-r--r--extra/mariabackup/ds_stdout.c2
-rw-r--r--extra/mariabackup/ds_tmpfile.c3
-rw-r--r--extra/mariabackup/encryption_plugin.cc157
-rw-r--r--extra/mariabackup/encryption_plugin.h7
-rw-r--r--extra/mariabackup/fil_cur.cc25
-rw-r--r--extra/mariabackup/fil_cur.h6
-rw-r--r--extra/mariabackup/innobackupex.cc27
-rw-r--r--extra/mariabackup/read_filt.cc6
-rw-r--r--extra/mariabackup/read_filt.h8
-rw-r--r--extra/mariabackup/version_check.pl1373
-rw-r--r--extra/mariabackup/write_filt.cc2
-rw-r--r--extra/mariabackup/write_filt.h3
-rw-r--r--extra/mariabackup/wsrep.cc7
-rw-r--r--extra/mariabackup/xb0xb.h78
-rw-r--r--extra/mariabackup/xb_regex.h43
-rw-r--r--extra/mariabackup/xbcrypt_common.c2
-rw-r--r--extra/mariabackup/xbcrypt_write.c2
-rw-r--r--extra/mariabackup/xbstream.c8
-rw-r--r--extra/mariabackup/xbstream_read.c8
-rw-r--r--extra/mariabackup/xbstream_write.c20
-rw-r--r--extra/mariabackup/xtrabackup.cc443
-rw-r--r--extra/mariabackup/xtrabackup.h30
-rw-r--r--extra/mariabackup/xtrabackup_version.h.in27
-rw-r--r--storage/xtradb/CMakeLists.txt3
38 files changed, 949 insertions, 3163 deletions
diff --git a/extra/mariabackup/CMakeLists.txt b/extra/mariabackup/CMakeLists.txt
index a84ac304486..f3369b49cee 100644
--- a/extra/mariabackup/CMakeLists.txt
+++ b/extra/mariabackup/CMakeLists.txt
@@ -13,65 +13,138 @@
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
-INCLUDE(gcrypt)
-INCLUDE(curl)
-INCLUDE(libev)
-ADD_SUBDIRECTORY(libarchive)
-ADD_SUBDIRECTORY(jsmn)
-
-FIND_GCRYPT()
-FIND_CURL()
-FIND_EV()
-
-# xxd is needed to embed version_check script
-FIND_PROGRAM(XXD_PATH xxd)
-
-IF(NOT XXD_PATH)
- MESSAGE(FATAL_ERROR "xxd not found. Try to install vim-common.")
-ENDIF(NOT XXD_PATH)
+OPTION(WITH_MARIABACKUP "Include mariabackup" ON)
+IF(NOT WITH_MARIABACKUP)
+ RETURN()
+ENDIF()
+
+
+IF(NOT WIN32)
+ CHECK_SYMBOL_EXISTS(regcomp regex.h HAVE_SYSTEM_REGEX)
+ IF(HAVE_SYSTEM_REGEX)
+ ADD_DEFINITIONS(-DHAVE_SYSTEM_REGEX)
+ ENDIF()
+ENDIF()
+
+IF(WITH_LIBARCHIVE STREQUAL "STATIC")
+ SET(CMAKE_FIND_LIBRARY_SUFFIXES .a .lib)
+ENDIF()
+
+FIND_PACKAGE(LibArchive)
+
+IF(NOT DEFINED WITH_LIBARCHIVE)
+ IF(LibArchive_FOUND)
+ SET(WITH_LIBARCHIVE_DEFAULT ON)
+ ELSE()
+ SET(WITH_LIBARCHIVE_DEFAULT OFF)
+ ENDIF()
+ SET(WITH_LIBARCHIVE ${WITH_LIBARCHIVE_DEFAULT} CACHE STRING "Use libarchive for streaming features (ON, OFF or STATIC)" )
+ENDIF()
+
+IF(NOT WITH_LIBARCHIVE MATCHES "^(ON|OFF|STATIC)$")
+ MESSAGE(FATAL_ERROR "Invalid value for WITH_LIBARCHIVE: '${WITH_LIBARCHIVE}'. Use one of ON, OFF or STATIC")
+ENDIF()
+
+IF(UNIX)
+ SET(PIC_FLAG -fPIC)
+ENDIF()
+
+IF((NOT WITH_LIBARCHIVE STREQUAL "OFF") AND (NOT LibArchive_FOUND))
+ IF(CMAKE_VERSION VERSION_LESS "2.8.12")
+ MESSAGE("libarchive can't be built, old cmake")
+ ELSE()
+ # Build a local version
+ INCLUDE(ExternalProject)
+ SET(libarchive_PREFIX ${CMAKE_CURRENT_BINARY_DIR}/libarchive)
+ SET(libarchive_CMAKE_ARGS
+ -DCMAKE_INSTALL_PREFIX:PATH=<INSTALL_DIR>
+ -DCMAKE_BUILD_TYPE=${CMAKE_BUILD_TYPE}
+ -DENABLE_ICONV=OFF
+ -DENABLE_TAR=ON
+ -DENABLE_OPENSSL=OFF
+ -DENABLE_TEST=OFF
+ "-DCMAKE_C_FLAGS_DEBUG=${CMAKE_C_FLAGS_DEBUG} ${PIC_FLAG}"
+ "-DCMAKE_C_FLAGS_RELWITHDEBINFO=${CMAKE_C_FLAGS_RELWITHDEBINFO} ${PIC_FLAG}"
+ "-DCMAKE_C_FLAGS_RELEASE=${CMAKE_C_FLAGS_RELEASE} ${PIC_FLAG}"
+ "-DCMAKE_C_FLAGS_MINSIZEREL=${CMAKE_C_FLAGS_MINSIZEREL} ${PIC_FLAG}"
+ )
+ IF(WIN32)
+ SET(libarchive_CMAKE_ARGS ${libarchive_CMAKE_ARGS} -DWINDOWS_VERSION=WIN7 -DCMAKE_DEBUG_POSTFIX=d)
+ ENDIF()
+
+ SET(LIBARCHIVE_DIR ${CMAKE_CURRENT_BINARY_DIR}/libarchive)
+ ExternalProject_Add(libarchive
+ PREFIX ${libarchive_PREFIX}
+ DOWNLOAD_DIR ${LIBARCHIVE_DIR}
+ URL http://www.libarchive.org/downloads/libarchive-3.2.2.tar.gz
+ INSTALL_DIR ${LIBARCHIVE_DIR}
+ CMAKE_ARGS ${libarchive_CMAKE_ARGS}
+ )
+ ADD_LIBRARY(archive_static STATIC IMPORTED)
+ ADD_DEPENDENCIES(archive_static libarchive)
+ IF(WIN32)
+ SET(LIBARCHIVE_RELEASE_LIB ${LIBARCHIVE_DIR}/lib/${CMAKE_STATIC_LIBRARY_PREFIX}archive_static${CMAKE_STATIC_LIBRARY_SUFFIX})
+ SET(LIBARCHIVE_DEBUG_LIB ${LIBARCHIVE_DIR}/lib/${CMAKE_STATIC_LIBRARY_PREFIX}archive_staticd${CMAKE_STATIC_LIBRARY_SUFFIX})
+ SET_PROPERTY(TARGET archive_static PROPERTY IMPORTED_LOCATION_RELWITHDEBINFO ${LIBARCHIVE_RELEASE_LIB})
+ SET_PROPERTY(TARGET archive_static PROPERTY IMPORTED_LOCATION_RELEASE ${LIBARCHIVE_RELEASE_LIB})
+ SET_PROPERTY(TARGET archive_static PROPERTY IMPORTED_LOCATION_DEBUG ${LIBARCHIVE_DEBUG_LIB})
+ SET_PROPERTY(TARGET archive_static PROPERTY IMPORTED_LOCATION_MINSIZEREL ${LIBARCHIVE_RELEASE_LIB})
+ ELSE()
+ SET_PROPERTY(TARGET archive_static PROPERTY IMPORTED_LOCATION ${LIBARCHIVE_DIR}/lib/${CMAKE_STATIC_LIBRARY_PREFIX}archive${CMAKE_STATIC_LIBRARY_SUFFIX})
+ ENDIF()
+
+ SET(LibArchive_FOUND ON )
+ SET(LibArchive_INCLUDE_DIRS ${LIBARCHIVE_DIR}/include )
+ SET(LibArchive_LIBRARIES archive_static)
+ IF(WIN32)
+ SET(LIBARCHIVE_STATIC 1)
+ ENDIF()
+ ENDIF()
+ENDIF()
+
+
+IF(WITH_LIBARCHIVE AND LibArchive_FOUND)
+ ADD_DEFINITIONS(-DHAVE_LIBARCHIVE)
+ IF(LIBARCHIVE_STATIC)
+ ADD_DEFINITIONS(-DLIBARCHIVE_STATIC)
+ ENDIF()
+ INCLUDE_DIRECTORIES(${LibArchive_INCLUDE_DIRS})
+ LINK_LIBRARIES(${LibArchive_LIBRARIES})
+ SET(DS_ARCHIVE_SOURCE ds_archive.c)
+ENDIF()
INCLUDE_DIRECTORIES(
${CMAKE_SOURCE_DIR}/include
- ${CMAKE_SOURCE_DIR}/storage/innobase/include
+ ${CMAKE_SOURCE_DIR}/storage/xtradb/include
${CMAKE_SOURCE_DIR}/sql
- ${CMAKE_SOURCE_DIR}/storage/innobase/xtrabackup/src/libarchive/libarchive
- ${CMAKE_SOURCE_DIR}/storage/innobase/xtrabackup/src/quicklz
- ${CMAKE_SOURCE_DIR}/storage/innobase/xtrabackup/src/jsmn
- ${GCRYPT_INCLUDE_DIR}
- ${CURL_INCLUDE_DIRS}
- ${LIBEV_INCLUDE_DIRS}
- ${CMAKE_CURRENT_BINARY_DIR}
+ ${CMAKE_CURRENT_SOURCE_DIR}/quicklz
+ ${CMAKE_CURRENT_SOURCE_DIR}
)
-ADD_DEFINITIONS(${SSL_DEFINES})
+IF(NOT HAVE_SYSTEM_REGEX)
+ INCLUDE_DIRECTORIES(${CMAKE_SOURCE_DIR}/pcre)
+ENDIF()
+ADD_DEFINITIONS(-UMYSQL_SERVER)
########################################################################
# xtrabackup binary
########################################################################
-CONFIGURE_FILE(${CMAKE_CURRENT_SOURCE_DIR}/xtrabackup_version.h.in
- ${CMAKE_CURRENT_BINARY_DIR}/xtrabackup_version.h )
-ADD_CUSTOM_COMMAND(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/version_check_pl.h
- COMMAND ${XXD_PATH} --include version_check.pl
- ${CMAKE_CURRENT_BINARY_DIR}/version_check_pl.h
- WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR})
+IF(WIN32)
+ SET(NT_SERVICE_SOURCE ${PROJECT_SOURCE_DIR}/sql/nt_servc.cc)
+ELSE()
+ SET(NT_SERVICE_SOURCE)
+ENDIF()
-ADD_CUSTOM_TARGET(GenVersionCheck
- DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/version_check_pl.h)
+ADD_DEFINITIONS(-DPCRE_STATIC=1)
-SET_SOURCE_FILES_PROPERTIES(
- xtrabackup.cc
- backup_mysql.cc
- PROPERTIES COMPILE_FLAGS -DMYSQL_CLIENT)
-
-MYSQL_ADD_EXECUTABLE(xtrabackup
+MYSQL_ADD_EXECUTABLE(mariabackup
xtrabackup.cc
innobackupex.cc
changed_page_bitmap.cc
- compact.cc
datasink.c
- ds_archive.c
+ ${DS_ARCHIVE_SOURCE}
ds_buffer.c
ds_compress.c
ds_encrypt.c
@@ -89,31 +162,29 @@ MYSQL_ADD_EXECUTABLE(xtrabackup
xbstream_write.c
backup_mysql.cc
backup_copy.cc
- ../../../../sql-common/client_authentication.cc
+ encryption_plugin.cc
+ ${PROJECT_SOURCE_DIR}/libmysql/libmysql.c
+ ${PROJECT_SOURCE_DIR}/sql/net_serv.cc
+ ${NT_SERVICE_SOURCE}
+ COMPONENT backup
)
-SET_TARGET_PROPERTIES(xtrabackup PROPERTIES ENABLE_EXPORTS TRUE)
-TARGET_LINK_LIBRARIES(xtrabackup
- mysqlserver
- ${GCRYPT_LIBS}
- archive_static
- )
+# Export all symbols on Unix, for better crash callstacks
+SET_TARGET_PROPERTIES(mariabackup PROPERTIES ENABLE_EXPORTS TRUE)
-ADD_DEPENDENCIES(xtrabackup GenVersionCheck)
-########################################################################
-# innobackupex symlink
-########################################################################
-ADD_CUSTOM_COMMAND(TARGET xtrabackup
- COMMAND ${CMAKE_COMMAND} ARGS -E create_symlink
- xtrabackup innobackupex)
-INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/innobackupex DESTINATION bin)
+TARGET_LINK_LIBRARIES(mariabackup sql)
+
+IF(NOT HAVE_SYSTEM_REGEX)
+ TARGET_LINK_LIBRARIES(mariabackup pcreposix)
+ENDIF()
+
########################################################################
# xbstream binary
########################################################################
-MYSQL_ADD_EXECUTABLE(xbstream
+MYSQL_ADD_EXECUTABLE(mbstream
ds_buffer.c
ds_local.c
ds_stdout.c
@@ -121,53 +192,15 @@ MYSQL_ADD_EXECUTABLE(xbstream
xbstream.c
xbstream_read.c
xbstream_write.c
- )
-
-SET_TARGET_PROPERTIES(xbstream
- PROPERTIES LINKER_LANGUAGE CXX
- )
-
-TARGET_LINK_LIBRARIES(xbstream
- mysys
- mysys_ssl
- )
-########################################################################
-# xbcrypt binary
-########################################################################
-MYSQL_ADD_EXECUTABLE(xbcrypt
- xbcrypt.c
- xbcrypt_common.c
- xbcrypt_read.c
- xbcrypt_write.c
+ COMPONENT backup
)
-SET_TARGET_PROPERTIES(xbcrypt
- PROPERTIES LINKER_LANGUAGE CXX
- )
-TARGET_LINK_LIBRARIES(xbcrypt
- ${GCRYPT_LIBS}
+TARGET_LINK_LIBRARIES(mbstream
mysys
- mysys_ssl
- )
+)
-########################################################################
-# xbcloud binary
-########################################################################
-MYSQL_ADD_EXECUTABLE(xbcloud
- xbcloud.cc
- )
-
-SET_TARGET_PROPERTIES(xbcloud
- PROPERTIES LINKER_LANGUAGE CXX
- )
-
-TARGET_LINK_LIBRARIES(xbcloud
- ${GCRYPT_LIBS}
- ${LIBEV_LIBRARIES}
- ${CURL_LIBRARIES}
- mysys
- mysys_ssl
- jsmn
- )
+IF(MSVC)
+ SET_TARGET_PROPERTIES(mbstream PROPERTIES LINK_FLAGS setargv.obj)
+ENDIF()
diff --git a/extra/mariabackup/backup_copy.cc b/extra/mariabackup/backup_copy.cc
index a051c64ee3b..478a6a66b97 100644
--- a/extra/mariabackup/backup_copy.cc
+++ b/extra/mariabackup/backup_copy.cc
@@ -1,6 +1,7 @@
/******************************************************
hot backup tool for InnoDB
(c) 2009-2015 Percona LLC and/or its affiliates
+(c) 2017 MariaDB
Originally Created 3/3/2009 Yasufumi Kinoshita
Written by Alexey Kopytov, Aleksandr Kuzminsky, Stewart Smith, Vadim Tkachenko,
Yasufumi Kinoshita, Ignacio Nin and Baron Schwartz.
@@ -48,13 +49,14 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#include <set>
#include <string>
#include <mysqld.h>
-#include <version_check_pl.h>
#include <sstream>
#include "fil_cur.h"
#include "xtrabackup.h"
#include "common.h"
#include "backup_copy.h"
#include "backup_mysql.h"
+#include <btr0btr.h>
+#include "xb0xb.h"
/* list of files to sync for --rsync mode */
@@ -110,6 +112,7 @@ struct datadir_thread_ctxt_t {
bool ret;
};
+static bool backup_files_from_datadir(const char *dir_path);
/************************************************************************
Retirn true if character if file separator */
@@ -225,7 +228,7 @@ datadir_iter_next_database(datadir_iter_t *it)
it->dbdir = NULL;
}
- while (fil_file_readdir_next_file(&it->err, it->datadir_path,
+ while (os_file_readdir_next_file(it->datadir_path,
it->dir, &it->dbinfo) == 0) {
ulint len;
@@ -340,7 +343,7 @@ datadir_iter_next_file(datadir_iter_t *it)
return(false);
}
- while (fil_file_readdir_next_file(&it->err, it->dbpath, it->dbdir,
+ while (os_file_readdir_next_file(it->dbpath, it->dbdir,
&it->fileinfo) == 0) {
if (it->fileinfo.type == OS_FILE_TYPE_DIR) {
@@ -449,9 +452,9 @@ struct datafile_cur_t {
uint thread_n;
byte* orig_buf;
byte* buf;
- ib_int64_t buf_size;
- ib_int64_t buf_read;
- ib_int64_t buf_offset;
+ size_t buf_size;
+ size_t buf_read;
+ size_t buf_offset;
};
static
@@ -486,7 +489,7 @@ datafile_open(const char *file, datafile_cur_t *cursor, uint thread_n)
cursor->abs_path,
OS_FILE_OPEN,
OS_FILE_READ_ONLY,
- &success);
+ &success, 0);
if (!success) {
/* The following call prints an error message */
os_file_get_last_error(TRUE);
@@ -498,7 +501,7 @@ datafile_open(const char *file, datafile_cur_t *cursor, uint thread_n)
return(false);
}
- if (my_fstat(cursor->file, &cursor->statinfo, MYF(MY_WME))) {
+ if (!my_stat(cursor->abs_path, &cursor->statinfo, 0)) {
msg("[%02u] error: cannot stat %s\n",
thread_n, cursor->abs_path);
@@ -510,7 +513,7 @@ datafile_open(const char *file, datafile_cur_t *cursor, uint thread_n)
posix_fadvise(cursor->file, 0, 0, POSIX_FADV_SEQUENTIAL);
cursor->buf_size = 10 * 1024 * 1024;
- cursor->buf = static_cast<byte *>(ut_malloc(cursor->buf_size));
+ cursor->buf = static_cast<byte *>(ut_malloc((ulint)cursor->buf_size));
return(true);
}
@@ -525,7 +528,7 @@ datafile_read(datafile_cur_t *cursor)
xtrabackup_io_throttling();
- to_read = min(cursor->statinfo.st_size - cursor->buf_offset,
+ to_read = (ulint)MY_MIN(cursor->statinfo.st_size - cursor->buf_offset,
cursor->buf_size);
if (to_read == 0) {
@@ -603,6 +606,11 @@ ends_with(const char *str, const char *suffix)
&& strcmp(str + str_len - suffix_len, suffix) == 0);
}
+static bool starts_with(const char *str, const char *prefix)
+{
+ return strncmp(str, prefix, strlen(prefix)) == 0;
+}
+
/************************************************************************
Create directories recursively.
@return 0 if directories created successfully. */
@@ -644,6 +652,7 @@ Return true if first and second arguments are the same path. */
bool
equal_paths(const char *first, const char *second)
{
+#ifdef HAVE_REALPATH
char real_first[PATH_MAX];
char real_second[PATH_MAX];
@@ -655,6 +664,9 @@ equal_paths(const char *first, const char *second)
}
return (strcmp(real_first, real_second) == 0);
+#else
+ return strcmp(first, second) == 0;
+#endif
}
/************************************************************************
@@ -675,10 +687,8 @@ directory_exists(const char *dir, bool create)
}
if (mkdirp(dir, 0777, MYF(0)) < 0) {
-
- msg("Can not create directory %s: %s\n", dir,
- my_strerror(errbuf, sizeof(errbuf), my_errno));
-
+ my_strerror(errbuf, sizeof(errbuf), my_errno);
+ msg("Can not create directory %s: %s\n", dir, errbuf);
return(false);
}
@@ -688,9 +698,9 @@ directory_exists(const char *dir, bool create)
os_dir = os_file_opendir(dir, FALSE);
if (os_dir == NULL) {
-
+ my_strerror(errbuf, sizeof(errbuf), my_errno);
msg("Can not open directory %s: %s\n", dir,
- my_strerror(errbuf, sizeof(errbuf), my_errno));
+ errbuf);
return(false);
}
@@ -1054,16 +1064,17 @@ move_file(ds_ctxt_t *datasink,
dst_file_path, thread_n);
msg_ts("[%02u] Removing %s\n", thread_n, src_file_path);
if (unlink(src_file_path) != 0) {
+ my_strerror(errbuf, sizeof(errbuf), errno);
msg("Error: unlink %s failed: %s\n",
src_file_path,
- my_strerror(errbuf,
- sizeof(errbuf), errno));
+ errbuf);
}
return(ret);
}
+ my_strerror(errbuf, sizeof(errbuf), my_errno);
msg("Can not move file %s to %s: %s\n",
src_file_path, dst_file_path_abs,
- my_strerror(errbuf, sizeof(errbuf), my_errno));
+ errbuf);
return(false);
}
@@ -1360,6 +1371,10 @@ backup_start()
return(false);
}
+ if (!backup_files_from_datadir(fil_path_to_mysql_datadir)) {
+ return false;
+ }
+
// There is no need to stop slave thread before coping non-Innodb data when
// --no-lock option is used because --no-lock option requires that no DDL or
// DML to non-transaction tables can occur.
@@ -1578,7 +1593,11 @@ ibx_cleanup_full_backup()
while (datadir_iter_next(it, &node)) {
if (node.is_empty_dir) {
+#ifdef _WIN32
+ DeleteFile(node.filepath);
+#else
rmdir(node.filepath);
+#endif
}
if (xtrabackup_incremental && !node.is_empty_dir
@@ -1605,6 +1624,9 @@ apply_log_finish()
return(true);
}
+extern void
+os_io_init_simple(void);
+
bool
copy_back()
{
@@ -1661,7 +1683,7 @@ copy_back()
}
srv_max_n_threads = 1000;
- os_sync_mutex = NULL;
+ //os_sync_mutex = NULL;
ut_mem_init();
/* temporally dummy value to avoid crash */
srv_page_size_shift = 14;
@@ -1764,10 +1786,9 @@ copy_back()
if (mkdirp(path, 0777, MYF(0)) < 0) {
char errbuf[MYSYS_STRERROR_SIZE];
-
+ my_strerror(errbuf, sizeof(errbuf), my_errno);
msg("Can not create directory %s: %s\n",
- path, my_strerror(errbuf,
- sizeof(errbuf), my_errno));
+ path, errbuf);
ret = false;
goto cleanup;
@@ -1855,13 +1876,12 @@ cleanup:
ds_data = NULL;
- sync_close();
- sync_initialized = FALSE;
- os_sync_free();
+ //os_sync_free();
mem_close();
- os_sync_mutex = NULL;
+ //os_sync_mutex = NULL;
ut_free_all_mem();
-
+ sync_close();
+ sync_initialized = FALSE;
return(ret);
}
@@ -1872,7 +1892,7 @@ decrypt_decompress_file(const char *filepath, uint thread_n)
char *dest_filepath = strdup(filepath);
bool needs_action = false;
- cmd << "cat " << filepath;
+ cmd << IF_WIN("type ","cat ") << filepath;
if (ends_with(filepath, ".xbcrypt") && opt_decrypt) {
cmd << " | xbcrypt --decrypt --encrypt-algo="
@@ -1926,7 +1946,7 @@ decrypt_decompress_file(const char *filepath, uint thread_n)
}
static
-os_thread_ret_t
+os_thread_ret_t STDCALL
decrypt_decompress_thread_func(void *arg)
{
bool ret = true;
@@ -1974,7 +1994,7 @@ decrypt_decompress()
datadir_iter_t *it = NULL;
srv_max_n_threads = 1000;
- os_sync_mutex = NULL;
+ //os_sync_mutex = NULL;
ut_mem_init();
os_sync_init();
sync_init();
@@ -2008,40 +2028,43 @@ decrypt_decompress()
sync_close();
sync_initialized = FALSE;
- os_sync_free();
- os_sync_mutex = NULL;
+ //os_sync_free();
+ //os_sync_mutex = NULL;
ut_free_all_mem();
return(ret);
}
-void
-version_check()
+/*
+ Copy some files from top level datadir.
+ Do not copy the Innodb files (ibdata1, redo log files),
+ as this is done in a separate step.
+*/
+static bool backup_files_from_datadir(const char *dir_path)
{
- if (opt_password != NULL) {
- setenv("option_mysql_password", opt_password, 1);
- }
- if (opt_user != NULL) {
- setenv("option_mysql_user", opt_user, 1);
- }
- if (opt_host != NULL) {
- setenv("option_mysql_host", opt_host, 1);
- }
- if (opt_socket != NULL) {
- setenv("option_mysql_socket", opt_socket, 1);
- }
- if (opt_port != 0) {
- char port[20];
- snprintf(port, sizeof(port), "%u", opt_port);
- setenv("option_mysql_port", port, 1);
- }
-
- FILE *pipe = popen("perl", "w");
- if (pipe == NULL) {
- return;
- }
+ os_file_dir_t dir = os_file_opendir(dir_path, TRUE);
+ os_file_stat_t info;
+ bool ret = true;
+ while (os_file_readdir_next_file(dir_path, dir, &info) == 0) {
- fputs((const char *)version_check_pl, pipe);
+ if (info.type != OS_FILE_TYPE_FILE)
+ continue;
- pclose(pipe);
+ const char *pname = strrchr(info.name, IF_WIN('\\', '/'));
+ if (!pname)
+ pname = info.name;
+
+ /* Copy aria log files, and aws keys for encryption plugins.*/
+ const char *prefixes[] = { "aria_log", "aws-kms-key" };
+ for (size_t i = 0; i < array_elements(prefixes); i++) {
+ if (starts_with(pname, prefixes[i])) {
+ ret = copy_file(ds_data, info.name, info.name, 1);
+ if (!ret) {
+ break;
+ }
+ }
+ }
+ }
+ os_file_closedir(dir);
+ return ret;
}
diff --git a/extra/mariabackup/backup_copy.h b/extra/mariabackup/backup_copy.h
index c8fc5fc8ba9..4b829982764 100644
--- a/extra/mariabackup/backup_copy.h
+++ b/extra/mariabackup/backup_copy.h
@@ -41,8 +41,6 @@ bool
copy_back();
bool
decrypt_decompress();
-void
-version_check();
bool
is_path_separator(char);
bool
diff --git a/extra/mariabackup/backup_mysql.cc b/extra/mariabackup/backup_mysql.cc
index 1ae8d10053e..ed5446b4806 100644
--- a/extra/mariabackup/backup_mysql.cc
+++ b/extra/mariabackup/backup_mysql.cc
@@ -38,6 +38,7 @@ this program; if not, write to the Free Software Foundation, Inc., 59 Temple
Place, Suite 330, Boston, MA 02111-1307 USA
*******************************************************/
+#define MYSQL_CLIENT
#include <my_global.h>
#include <mysql.h>
@@ -47,10 +48,12 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#include <limits>
#include "common.h"
#include "xtrabackup.h"
-#include "xtrabackup_version.h"
+#include "mysql_version.h"
#include "backup_copy.h"
#include "backup_mysql.h"
#include "mysqld.h"
+#include "encryption_plugin.h"
+#include <sstream>
char *tool_name;
@@ -88,15 +91,7 @@ time_t history_lock_time;
MYSQL *mysql_connection;
-extern "C" {
-MYSQL * STDCALL
-cli_mysql_real_connect(MYSQL *mysql,const char *host, const char *user,
- const char *passwd, const char *db,
- uint port, const char *unix_socket,ulong client_flag);
-}
-
-#define mysql_real_connect cli_mysql_real_connect
-
+my_bool opt_ssl_verify_server_cert;
MYSQL *
xb_mysql_connect()
@@ -136,11 +131,6 @@ xb_mysql_connect()
}
mysql_options(connection,MYSQL_OPT_SSL_VERIFY_SERVER_CERT,
(char*)&opt_ssl_verify_server_cert);
-#if !defined(HAVE_YASSL)
- if (opt_server_public_key && *opt_server_public_key)
- mysql_options(connection, MYSQL_SERVER_PUBLIC_KEY,
- opt_server_public_key);
-#endif
#endif
if (!mysql_real_connect(connection,
@@ -310,7 +300,7 @@ check_server_version(unsigned long version_number,
version_supported = version_supported
|| (version_number > 50500 && version_number < 50700);
version_supported = version_supported
- || ((version_number > 100000 && version_number < 100300)
+ || ((version_number > 100000)
&& server_flavor == FLAVOR_MARIADB);
if (mysql51 && innodb_version == NULL) {
@@ -596,7 +586,7 @@ select_incremental_lsn_from_history(lsn_t *incremental_lsn)
if (opt_incremental_history_name) {
mysql_real_escape_string(mysql_connection, buf,
opt_incremental_history_name,
- strlen(opt_incremental_history_name));
+ (unsigned long)strlen(opt_incremental_history_name));
ut_snprintf(query, sizeof(query),
"SELECT innodb_to_lsn "
"FROM PERCONA_SCHEMA.xtrabackup_history "
@@ -609,7 +599,7 @@ select_incremental_lsn_from_history(lsn_t *incremental_lsn)
if (opt_incremental_history_uuid) {
mysql_real_escape_string(mysql_connection, buf,
opt_incremental_history_uuid,
- strlen(opt_incremental_history_uuid));
+ (unsigned long)strlen(opt_incremental_history_uuid));
ut_snprintf(query, sizeof(query),
"SELECT innodb_to_lsn "
"FROM PERCONA_SCHEMA.xtrabackup_history "
@@ -756,7 +746,7 @@ have_queries_to_wait_for(MYSQL *connection, uint threshold)
static
void
-kill_long_queries(MYSQL *connection, uint timeout)
+kill_long_queries(MYSQL *connection, time_t timeout)
{
MYSQL_RES *result;
MYSQL_ROW row;
@@ -768,15 +758,15 @@ kill_long_queries(MYSQL *connection, uint timeout)
all_queries = (opt_kill_long_query_type == QUERY_TYPE_ALL);
while ((row = mysql_fetch_row(result)) != NULL) {
const char *info = row[7];
- int duration = atoi(row[5]);
+ long long duration = atoll(row[5]);
char *id = row[0];
if (info != NULL &&
- duration >= (int)timeout &&
+ (time_t)duration >= timeout &&
((all_queries && is_query(info)) ||
is_select_query(info))) {
msg_ts("Killing query %s (duration %d sec): %s\n",
- id, duration, info);
+ id, (int)duration, info);
ut_snprintf(kill_stmt, sizeof(kill_stmt),
"KILL %s", id);
xb_mysql_query(connection, kill_stmt, false, false);
@@ -1378,7 +1368,18 @@ cleanup:
return(result);
}
-
+static string escape_and_quote(MYSQL *mysql,const char *str)
+{
+ if (!str)
+ return "NULL";
+ size_t len = strlen(str);
+ char* escaped = (char *)alloca(2 * len + 3);
+ escaped[0] = '\'';
+ size_t new_len = mysql_real_escape_string(mysql, escaped+1, str, len);
+ escaped[new_len + 1] = '\'';
+ escaped[new_len + 2] = 0;
+ return string(escaped);
+}
/*********************************************************************//**
Writes xtrabackup_info file and if backup_history is enable creates
@@ -1388,25 +1389,16 @@ backup. */
bool
write_xtrabackup_info(MYSQL *connection)
{
- MYSQL_STMT *stmt;
- MYSQL_BIND bind[19];
+
char *uuid = NULL;
char *server_version = NULL;
char buf_start_time[100];
char buf_end_time[100];
- int idx;
tm tm;
my_bool null = TRUE;
-
+ ostringstream oss;
const char *xb_stream_name[] = {"file", "tar", "xbstream"};
- const char *ins_query = "insert into PERCONA_SCHEMA.xtrabackup_history("
- "uuid, name, tool_name, tool_command, tool_version, "
- "ibbackup_version, server_version, start_time, end_time, "
- "lock_time, binlog_pos, innodb_from_lsn, innodb_to_lsn, "
- "partial, incremental, format, compact, compressed, "
- "encrypted) "
- "values(?,?,?,?,?,?,?,from_unixtime(?),from_unixtime(?),"
- "?,?,?,?,?,?,?,?,?,?)";
+
ut_ad(xtrabackup_stream_fmt < 3);
@@ -1419,6 +1411,11 @@ write_xtrabackup_info(MYSQL *connection)
localtime_r(&history_end_time, &tm);
strftime(buf_end_time, sizeof(buf_end_time),
"%Y-%m-%d %H:%M:%S", &tm);
+ bool is_partial = (xtrabackup_tables
+ || xtrabackup_tables_file
+ || xtrabackup_databases
+ || xtrabackup_databases_file);
+
backup_file_printf(XTRABACKUP_INFO,
"uuid = %s\n"
"name = %s\n"
@@ -1443,23 +1440,20 @@ write_xtrabackup_info(MYSQL *connection)
opt_history ? opt_history : "", /* name */
tool_name, /* tool_name */
tool_args, /* tool_command */
- XTRABACKUP_VERSION, /* tool_version */
- XTRABACKUP_VERSION, /* ibbackup_version */
+ MYSQL_SERVER_VERSION, /* tool_version */
+ MYSQL_SERVER_VERSION, /* ibbackup_version */
server_version, /* server_version */
buf_start_time, /* start_time */
buf_end_time, /* end_time */
- history_lock_time, /* lock_time */
+ (int)history_lock_time, /* lock_time */
mysql_binlog_position ?
mysql_binlog_position : "", /* binlog_pos */
incremental_lsn, /* innodb_from_lsn */
metadata_to_lsn, /* innodb_to_lsn */
- (xtrabackup_tables /* partial */
- || xtrabackup_tables_file
- || xtrabackup_databases
- || xtrabackup_databases_file) ? "Y" : "N",
+ is_partial? "Y" : "N",
xtrabackup_incremental ? "Y" : "N", /* incremental */
xb_stream_name[xtrabackup_stream_fmt], /* format */
- xtrabackup_compact ? "Y" : "N", /* compact */
+ "N", /* compact */
xtrabackup_compress ? "compressed" : "N", /* compressed */
xtrabackup_encrypt ? "Y" : "N"); /* encrypted */
@@ -1492,142 +1486,36 @@ write_xtrabackup_info(MYSQL *connection)
"encrypted ENUM('Y', 'N') DEFAULT NULL"
") CHARACTER SET utf8 ENGINE=innodb", false);
- stmt = mysql_stmt_init(connection);
-
- mysql_stmt_prepare(stmt, ins_query, strlen(ins_query));
-
- memset(bind, 0, sizeof(bind));
- idx = 0;
-
- /* uuid */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = uuid;
- bind[idx].buffer_length = strlen(uuid);
- ++idx;
-
- /* name */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(opt_history);
- bind[idx].buffer_length = strlen(opt_history);
- if (!(opt_history && *opt_history)) {
- bind[idx].is_null = &null;
- }
- ++idx;
-
- /* tool_name */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = tool_name;
- bind[idx].buffer_length = strlen(tool_name);
- ++idx;
-
- /* tool_command */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = tool_args;
- bind[idx].buffer_length = strlen(tool_args);
- ++idx;
-
- /* tool_version */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(XTRABACKUP_VERSION);
- bind[idx].buffer_length = strlen(XTRABACKUP_VERSION);
- ++idx;
-
- /* ibbackup_version */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(XTRABACKUP_VERSION);
- bind[idx].buffer_length = strlen(XTRABACKUP_VERSION);
- ++idx;
-
- /* server_version */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = server_version;
- bind[idx].buffer_length = strlen(server_version);
- ++idx;
-
- /* start_time */
- bind[idx].buffer_type = MYSQL_TYPE_LONG;
- bind[idx].buffer = &history_start_time;
- ++idx;
-
- /* end_time */
- bind[idx].buffer_type = MYSQL_TYPE_LONG;
- bind[idx].buffer = &history_end_time;
- ++idx;
-
- /* lock_time */
- bind[idx].buffer_type = MYSQL_TYPE_LONG;
- bind[idx].buffer = &history_lock_time;
- ++idx;
-
- /* binlog_pos */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = mysql_binlog_position;
- if (mysql_binlog_position != NULL) {
- bind[idx].buffer_length = strlen(mysql_binlog_position);
- } else {
- bind[idx].is_null = &null;
- }
- ++idx;
-
- /* innodb_from_lsn */
- bind[idx].buffer_type = MYSQL_TYPE_LONGLONG;
- bind[idx].buffer = (char*)(&incremental_lsn);
- ++idx;
-
- /* innodb_to_lsn */
- bind[idx].buffer_type = MYSQL_TYPE_LONGLONG;
- bind[idx].buffer = (char*)(&metadata_to_lsn);
- ++idx;
-
- /* partial (Y | N) */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)((xtrabackup_tables
- || xtrabackup_tables_file
- || xtrabackup_databases
- || xtrabackup_databases_file) ? "Y" : "N");
- bind[idx].buffer_length = 1;
- ++idx;
-
- /* incremental (Y | N) */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(
- (xtrabackup_incremental
- || xtrabackup_incremental_basedir
- || opt_incremental_history_name
- || opt_incremental_history_uuid) ? "Y" : "N");
- bind[idx].buffer_length = 1;
- ++idx;
-
- /* format (file | tar | xbstream) */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(xb_stream_name[xtrabackup_stream_fmt]);
- bind[idx].buffer_length = strlen(xb_stream_name[xtrabackup_stream_fmt]);
- ++idx;
-
- /* compact (Y | N) */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(xtrabackup_compact ? "Y" : "N");
- bind[idx].buffer_length = 1;
- ++idx;
-
- /* compressed (Y | N) */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(xtrabackup_compress ? "Y" : "N");
- bind[idx].buffer_length = 1;
- ++idx;
-
- /* encrypted (Y | N) */
- bind[idx].buffer_type = MYSQL_TYPE_STRING;
- bind[idx].buffer = (char*)(xtrabackup_encrypt ? "Y" : "N");
- bind[idx].buffer_length = 1;
- ++idx;
-
- ut_ad(idx == 19);
-
- mysql_stmt_bind_param(stmt, bind);
-
- mysql_stmt_execute(stmt);
- mysql_stmt_close(stmt);
+
+#define ESCAPE_BOOL(expr) ((expr)?"'Y'":"'N'")
+
+ oss << "insert into PERCONA_SCHEMA.xtrabackup_history("
+ << "uuid, name, tool_name, tool_command, tool_version,"
+ << "ibbackup_version, server_version, start_time, end_time,"
+ << "lock_time, binlog_pos, innodb_from_lsn, innodb_to_lsn,"
+ << "partial, incremental, format, compact, compressed, "
+ << "encrypted) values("
+ << escape_and_quote(connection, uuid) << ","
+ << escape_and_quote(connection, opt_history) << ","
+ << escape_and_quote(connection, tool_name) << ","
+ << escape_and_quote(connection, tool_args) << ","
+ << escape_and_quote(connection, MYSQL_SERVER_VERSION) << ","
+ << escape_and_quote(connection, MYSQL_SERVER_VERSION) << ","
+ << escape_and_quote(connection, server_version) << ","
+ << "from_unixtime(" << history_start_time << "),"
+ << "from_unixtime(" << history_end_time << "),"
+ << history_lock_time << ","
+ << escape_and_quote(connection, mysql_binlog_position) << ","
+ << incremental_lsn << ","
+ << metadata_to_lsn << ","
+ << ESCAPE_BOOL(is_partial) << ","
+ << ESCAPE_BOOL(xtrabackup_incremental)<< ","
+ << escape_and_quote(connection,xb_stream_name[xtrabackup_stream_fmt]) <<","
+ << ESCAPE_BOOL(false) << ","
+ << ESCAPE_BOOL(xtrabackup_compress) << ","
+ << ESCAPE_BOOL(xtrabackup_encrypt) <<")";
+
+ xb_mysql_query(mysql_connection, oss.str().c_str(), false);
cleanup:
@@ -1637,10 +1525,11 @@ cleanup:
return(true);
}
-bool
-write_backup_config_file()
+extern const char *innodb_checksum_algorithm_names[];
+
+bool write_backup_config_file()
{
- return backup_file_printf("backup-my.cnf",
+ int rc= backup_file_printf("backup-my.cnf",
"# This MySQL options file was generated by innobackupex.\n\n"
"# The MySQL server\n"
"[mysqld]\n"
@@ -1649,19 +1538,18 @@ write_backup_config_file()
"innodb_data_file_path=%s\n"
"innodb_log_files_in_group=%lu\n"
"innodb_log_file_size=%lld\n"
- "innodb_fast_checksum=%s\n"
"innodb_page_size=%lu\n"
"innodb_log_block_size=%lu\n"
"innodb_undo_directory=%s\n"
"innodb_undo_tablespaces=%lu\n"
"%s%s\n"
- "%s%s\n",
+ "%s%s\n"
+ "%s\n",
innodb_checksum_algorithm_names[srv_checksum_algorithm],
innodb_checksum_algorithm_names[srv_log_checksum_algorithm],
innobase_data_file_path,
srv_n_log_files,
innobase_log_file_size,
- srv_fast_checksum ? "true" : "false",
srv_page_size,
srv_log_block_size,
srv_undo_dir,
@@ -1671,7 +1559,9 @@ write_backup_config_file()
innobase_buffer_pool_filename ?
"innodb_buffer_pool_filename=" : "",
innobase_buffer_pool_filename ?
- innobase_buffer_pool_filename : "");
+ innobase_buffer_pool_filename : "",
+ encryption_plugin_get_config());
+ return rc;
}
diff --git a/extra/mariabackup/changed_page_bitmap.cc b/extra/mariabackup/changed_page_bitmap.cc
index e385474c7aa..435b7fb6172 100644
--- a/extra/mariabackup/changed_page_bitmap.cc
+++ b/extra/mariabackup/changed_page_bitmap.cc
@@ -447,7 +447,7 @@ log_online_open_bitmap_file_read_only(
= os_file_create_simple_no_error_handling(0, bitmap_file->name,
OS_FILE_OPEN,
OS_FILE_READ_ONLY,
- &success);
+ &success,0);
if (UNIV_UNLIKELY(!success)) {
/* Here and below assume that bitmap file names do not
diff --git a/extra/mariabackup/common.h b/extra/mariabackup/common.h
index abbc4b41a85..7b1dfd7a0db 100644
--- a/extra/mariabackup/common.h
+++ b/extra/mariabackup/common.h
@@ -26,6 +26,46 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
#include <fcntl.h>
#include <stdarg.h>
+
+# define fil_is_user_tablespace_id(i) ((i) > srv_undo_tablespaces_open)
+
+#ifdef _MSC_VER
+#define stat _stati64
+#define PATH_MAX MAX_PATH
+#endif
+
+#ifndef HAVE_VASPRINTF
+static inline int vasprintf(char **strp, const char *fmt, va_list args)
+{
+ int len;
+#ifdef _MSC_VER
+ len = _vscprintf(fmt, args);
+#else
+ len = vsnprintf(NULL, 0, fmt, args);
+#endif
+ if (len < 0)
+ {
+ return -1;
+ }
+ *strp = (char *)malloc(len + 1);
+ if (!*strp)
+ {
+ return -1;
+ }
+ vsprintf(*strp, fmt, args);
+ return len;
+}
+
+static inline int asprintf(char **strp, const char *fmt,...)
+{
+ va_list args;
+ va_start(args, fmt);
+ int len = vasprintf(strp, fmt, args);
+ va_end(args);
+ return len;
+}
+#endif
+
#define xb_a(expr) \
do { \
if (!(expr)) { \
@@ -93,15 +133,15 @@ static inline int msg_ts(const char *fmt, ...)
/***********************************************************************
Computes bit shift for a given value. If the argument is not a power
of 2, returns 0.*/
-static inline ulong
-get_bit_shift(ulong value)
+static inline size_t
+get_bit_shift(size_t value)
{
- ulong shift;
+ size_t shift;
if (value == 0)
return 0;
- for (shift = 0; !(value & 1UL); shift++) {
+ for (shift = 0; !(value & 1); shift++) {
value >>= 1;
}
return (value >> 1) ? 0 : shift;
diff --git a/extra/mariabackup/compact.cc b/extra/mariabackup/compact.cc
deleted file mode 100644
index 5d08a6e02b2..00000000000
--- a/extra/mariabackup/compact.cc
+++ /dev/null
@@ -1,1059 +0,0 @@
-/******************************************************
-XtraBackup: hot backup tool for InnoDB
-(c) 2009-2014 Percona LLC and/or its affiliates.
-Originally Created 3/3/2009 Yasufumi Kinoshita
-Written by Alexey Kopytov, Aleksandr Kuzminsky, Stewart Smith, Vadim Tkachenko,
-Yasufumi Kinoshita, Ignacio Nin and Baron Schwartz.
-
-This program is free software; you can redistribute it and/or modify
-it under the terms of the GNU General Public License as published by
-the Free Software Foundation; version 2 of the License.
-
-This program is distributed in the hope that it will be useful,
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU General Public License for more details.
-
-You should have received a copy of the GNU General Public License
-along with this program; if not, write to the Free Software
-Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
-
-*******************************************************/
-
-/* Compact backups implementation */
-
-#include <my_base.h>
-#include <table.h>
-
-#include <univ.i>
-#include <dict0mem.h>
-#include <dict0priv.h>
-#include <fsp0fsp.h>
-#include <handler0alter.h>
-#include <ibuf0ibuf.h>
-#include <page0page.h>
-#include <row0merge.h>
-#include "common.h"
-#include "write_filt.h"
-#include "fil_cur.h"
-#include "xtrabackup.h"
-#include "ds_buffer.h"
-#include "xb0xb.h"
-
-/* Number of the first primary key page in an .ibd file */
-#define XB_FIRST_CLUSTERED_INDEX_PAGE_NO 3
-
-/* Suffix for page map files */
-#define XB_PAGE_MAP_SUFFIX ".pmap"
-#define XB_TMPFILE_SUFFIX ".tmp"
-
-/* Page range */
-struct page_range_t {
- ulint from; /*!< range start */
- ulint to; /*!< range end */
-};
-
-/* Cursor in a page map file */
-struct page_map_cursor_t {
- File fd; /*!< file descriptor */
- IO_CACHE cache; /*!< IO_CACHE associated with fd */
-};
-
-/* Table descriptor for the index rebuild operation */
-struct index_rebuild_table_t {
- char* name; /* table name */
- ulint space_id; /* space ID */
- UT_LIST_NODE_T(index_rebuild_table_t) list; /* list node */
-};
-
-/* Thread descriptor for the index rebuild operation */
-struct index_rebuild_thread_t {
- ulint num; /* thread number */
- pthread_t id; /* thread ID */
-};
-
-/* Empty page use to replace skipped pages in the data files */
-static byte empty_page[UNIV_PAGE_SIZE_MAX];
-static const char compacted_page_magic[] = "COMPACTP";
-static const size_t compacted_page_magic_size =
- sizeof(compacted_page_magic) - 1;
-static const ulint compacted_page_magic_offset = FIL_PAGE_DATA;
-
-/* Mutex protecting table_list */
-static pthread_mutex_t table_list_mutex;
-/* List of tablespaces to process by the index rebuild operation */
-static UT_LIST_BASE_NODE_T(index_rebuild_table_t) table_list;
-
-
-/************************************************************************
-Compact page filter. */
-static my_bool wf_compact_init(xb_write_filt_ctxt_t *ctxt, char *dst_name,
- xb_fil_cur_t *cursor);
-static my_bool wf_compact_process(xb_write_filt_ctxt_t *ctxt,
- ds_file_t *dstfile);
-static my_bool wf_compact_finalize(xb_write_filt_ctxt_t *ctxt,
- ds_file_t *dstfile);
-xb_write_filt_t wf_compact = {
- &wf_compact_init,
- &wf_compact_process,
- &wf_compact_finalize,
- NULL
-};
-
-/************************************************************************
-Initialize the compact page filter.
-
-@return TRUE on success, FALSE on error. */
-static my_bool
-wf_compact_init(xb_write_filt_ctxt_t *ctxt,
- char *dst_name __attribute__((unused)), xb_fil_cur_t *cursor)
-{
- xb_wf_compact_ctxt_t *cp = &(ctxt->u.wf_compact_ctxt);
- char page_map_name[FN_REFLEN];
- MY_STAT mystat;
-
- ctxt->cursor = cursor;
- cp->clustered_index_found = FALSE;
- cp->inside_skipped_range = FALSE;
- cp->free_limit = 0;
-
- /* Don't compact the system table space */
- cp->skip = cursor->is_system;
- if (cp->skip) {
- return(TRUE);
- }
-
- snprintf(page_map_name, sizeof(page_map_name), "%s%s", dst_name,
- XB_PAGE_MAP_SUFFIX);
-
- cp->ds_buffer = ds_create(xtrabackup_target_dir, DS_TYPE_BUFFER);
- if (cp->ds_buffer == NULL) {
- return(FALSE);
- }
-
- ds_set_pipe(cp->ds_buffer, ds_meta);
-
- memset(&mystat, 0, sizeof(mystat));
- mystat.st_mtime = my_time(0);
- cp->buffer = ds_open(cp->ds_buffer, page_map_name, &mystat);
- if (cp->buffer == NULL) {
- msg("xtrabackup: Error: cannot open output stream for %s\n",
- page_map_name);
- return(FALSE);
- }
-
- return(TRUE);
-}
-
-/************************************************************************
-Check if the specified page should be skipped. We currently skip all
-non-clustered index pages for compact backups.
-
-@return TRUE if the page should be skipped. */
-static my_bool
-check_if_skip_page(xb_wf_compact_ctxt_t *cp, xb_fil_cur_t *cursor, ulint offset)
-{
- byte *page;
- ulint page_no;
- ulint page_type;
- index_id_t index_id;
-
-
- xb_ad(cursor->is_system == FALSE);
-
- page = cursor->buf + cursor->page_size * offset;
- page_no = cursor->buf_page_no + offset;
- page_type = fil_page_get_type(page);
-
- if (UNIV_UNLIKELY(page_no == 0)) {
-
- cp->free_limit = mach_read_from_4(page + FSP_HEADER_OFFSET +
- FSP_FREE_LIMIT);
- } else if (UNIV_UNLIKELY(page_no == XB_FIRST_CLUSTERED_INDEX_PAGE_NO)) {
-
- xb_ad(cp->clustered_index_found == FALSE);
-
- if (page_type != FIL_PAGE_INDEX) {
-
- /* Uninitialized clustered index root page, there's
- nothing we can do to compact the space.*/
-
- msg("[%02u] Uninitialized page type value (%lu) in the "
- "clustered index root page of tablespace %s. "
- "Will not be compacted.\n",
- cursor->thread_n,
- page_type, cursor->rel_path);
-
- cp->skip = TRUE;
-
- return(FALSE);
- }
-
- cp->clustered_index =
- mach_read_from_8(page + PAGE_HEADER + PAGE_INDEX_ID);
- cp->clustered_index_found = TRUE;
- } else if (UNIV_UNLIKELY(page_no >= cp->free_limit)) {
-
- /* Skip unused pages above free limit, if that value is set in
- the FSP header.*/
-
- return(cp->free_limit > 0);
- } else if (cp->clustered_index_found && page_type == FIL_PAGE_INDEX) {
-
- index_id = mach_read_from_8(page + PAGE_HEADER + PAGE_INDEX_ID);
- if (index_id != cp->clustered_index) {
-
- ulint fseg_hdr_space =
- mach_read_from_4(page + PAGE_HEADER +
- PAGE_BTR_SEG_TOP);
- ulint fseg_hdr_page_no =
- mach_read_from_4(page + PAGE_HEADER +
- PAGE_BTR_SEG_TOP + 4);
- ulint fseg_hdr_offset =
- mach_read_from_2(page + PAGE_HEADER +
- PAGE_BTR_SEG_TOP + 8);
-
- /* Don't skip root index pages, i.e. the ones where the
- above fields are defined. We need root index pages to be
- able to correctly drop the indexes later, as they
- contain fseg inode pointers. */
-
- return(fseg_hdr_space == 0 &&
- fseg_hdr_page_no == 0 &&
- fseg_hdr_offset == 0);
- }
- }
-
- return(FALSE);
-}
-
-/************************************************************************
-Run the next batch of pages through the compact page filter.
-
-@return TRUE on success, FALSE on error. */
-static my_bool
-wf_compact_process(xb_write_filt_ctxt_t *ctxt, ds_file_t *dstfile)
-{
- xb_fil_cur_t *cursor = ctxt->cursor;
- ulint page_size = cursor->page_size;
- byte *page;
- byte *buf_end;
- byte *write_from;
- xb_wf_compact_ctxt_t *cp = &(ctxt->u.wf_compact_ctxt);
- ulint i;
- ulint page_no;
- byte tmp[4];
-
- if (cp->skip) {
- return(!ds_write(dstfile, cursor->buf, cursor->buf_read));
- }
-
- write_from = NULL;
- buf_end = cursor->buf + cursor->buf_read;
- for (i = 0, page = cursor->buf; page < buf_end;
- i++, page += page_size) {
-
- page_no = cursor->buf_page_no + i;
-
- if (!check_if_skip_page(cp, cursor, i)) {
-
- if (write_from == NULL) {
- write_from = page;
- }
-
- if (cp->inside_skipped_range) {
- cp->inside_skipped_range = FALSE;
-
- /* Write the last range endpoint to the
- skipped pages map */
-
- xb_ad(page_no > 0);
- mach_write_to_4(tmp, page_no - 1);
- if (ds_write(cp->buffer, tmp, sizeof(tmp))) {
- return(FALSE);
- }
- }
- continue;
- }
-
- if (write_from != NULL) {
-
- /* The first skipped page in this block, write the
- non-skipped ones to the data file */
-
- if (ds_write(dstfile, write_from, page - write_from)) {
- return(FALSE);
- }
-
- write_from = NULL;
- }
-
- if (!cp->inside_skipped_range) {
-
- /* The first skipped page in range, write the first
- range endpoint to the skipped pages map */
-
- cp->inside_skipped_range = TRUE;
-
- mach_write_to_4(tmp, page_no);
- if (ds_write(cp->buffer, tmp, sizeof(tmp))) {
- return(FALSE);
- }
- }
- }
-
- /* Write the remaining pages in the buffer, if any */
- if (write_from != NULL &&
- ds_write(dstfile, write_from, buf_end - write_from)) {
- return(FALSE);
- }
-
- return(TRUE);
-}
-
-/************************************************************************
-Close the compact filter's page map stream.
-
-@return TRUE on success, FALSE on error. */
-static my_bool
-wf_compact_finalize(xb_write_filt_ctxt_t *ctxt,
- ds_file_t *dstfile __attribute__((unused)))
-{
- xb_fil_cur_t *cursor = ctxt->cursor;
- xb_wf_compact_ctxt_t *cp = &(ctxt->u.wf_compact_ctxt);
- my_bool rc = TRUE;
-
- /* Write the last endpoint of the current range, if the last pages of
- the space have been skipped. */
- if (cp->inside_skipped_range) {
- byte tmp[4];
-
- mach_write_to_4(tmp, cursor->space_size - 1);
- if (ds_write(cp->buffer, tmp, sizeof(tmp))) {
- return(FALSE);
- }
-
- cp->inside_skipped_range = FALSE;
- }
-
- if (cp->buffer) {
- if (ds_close(cp->buffer)) {
- rc = FALSE;
- }
- }
- if (cp->ds_buffer) {
- ds_destroy(cp->ds_buffer);
- }
-
- return(rc);
-}
-
-/************************************************************************
-Open a page map file and return a cursor.
-
-@return page map cursor, or NULL if the file doesn't exist. */
-static page_map_cursor_t *
-page_map_file_open(const char *path)
-{
- MY_STAT statinfo;
- page_map_cursor_t *pmap_cur;
- int rc;
-
- if (my_stat(path, &statinfo, MYF(0)) == NULL) {
-
- return(NULL);
- }
-
- /* The maximum possible page map file corresponds to a 64 TB tablespace
- and the worst case when every other page was skipped. That is, 2^32/2
- page ranges = 16 GB. */
- xb_a(statinfo.st_size < (off_t) 16 * 1024 * 1024 * 1024);
-
- /* Must be a series of 8-byte tuples */
- xb_a(statinfo.st_size % 8 == 0);
-
- pmap_cur = (page_map_cursor_t *) my_malloc(sizeof(page_map_cursor_t),
- MYF(MY_FAE));
-
- pmap_cur->fd = my_open(path, O_RDONLY, MYF(MY_WME));
- xb_a(pmap_cur->fd != 0);
-
- rc = init_io_cache(&pmap_cur->cache, pmap_cur->fd, 0, READ_CACHE,
- 0, 0, MYF(MY_WME));
- xb_a(rc == 0);
-
- return(pmap_cur);
-}
-
-/************************************************************************
-Read the next range from a page map file and update the cursor.
-
-@return TRUE on success, FALSE on end-of-file. */
-static ibool
-page_map_file_next(page_map_cursor_t *pmap_cur, page_range_t *range)
-{
- byte buf[8];
-
- xb_ad(pmap_cur != NULL);
-
- if (my_b_read(&pmap_cur->cache, buf, sizeof(buf))) {
- return(FALSE);
- }
-
- range->from = mach_read_from_4(buf);
- range->to = mach_read_from_4(buf + 4);
-
- return(TRUE);
-}
-
-/************************************************************************
-Close the page map cursor.*/
-static void
-page_map_file_close(page_map_cursor_t *pmap_cur)
-{
- int rc;
-
- xb_ad(pmap_cur != NULL);
-
- rc = end_io_cache(&pmap_cur->cache);
- xb_a(rc == 0);
-
- posix_fadvise(pmap_cur->fd, 0, 0, POSIX_FADV_DONTNEED);
-
- rc = my_close(pmap_cur->fd, MY_WME);
- xb_a(rc == 0);
-
- my_free(pmap_cur);
-}
-
-/****************************************************************************
-Expand a single data file according to the skipped pages maps created by
---compact.
-
-@return TRUE on success, FALSE on failure. */
-static my_bool
-xb_expand_file(fil_node_t *node)
-{
- char pmapfile_path[FN_REFLEN];
- char tmpfile_path[FN_REFLEN];
- xb_fil_cur_t cursor;
- xb_fil_cur_result_t res;
- ds_ctxt_t *ds_local;
- ds_ctxt_t *ds_buffer;
- ds_file_t *tmpfile;
- my_bool success = FALSE;
- ulint i;
- byte *page;
- ulint page_expected_no;
- page_map_cursor_t *pmap_cur;
- ibool have_next_range;
- page_range_t pmap_range;
-
- xb_ad(trx_sys_sys_space(node->space->id) == FALSE);
-
- snprintf(pmapfile_path, sizeof(pmapfile_path), "%s%s",
- node->name, XB_PAGE_MAP_SUFFIX);
-
- /* Skip files that don't have a corresponding page map file */
-
- if (!(pmap_cur = page_map_file_open(pmapfile_path))) {
-
- msg("Not expanding %s\n", node->name);
-
- return(FALSE);
- }
-
- msg("Expanding %s\n", node->name);
-
- ds_local = ds_create(".", DS_TYPE_LOCAL);
- ds_buffer = ds_create(".", DS_TYPE_BUFFER);
-
- xb_a(ds_local != NULL && ds_buffer != NULL);
-
- ds_buffer_set_size(ds_buffer, FSP_EXTENT_SIZE * UNIV_PAGE_SIZE_MAX);
-
- ds_set_pipe(ds_buffer, ds_local);
-
- res = xb_fil_cur_open(&cursor, &rf_pass_through, node, 1);
- xb_a(res == XB_FIL_CUR_SUCCESS);
-
- snprintf(tmpfile_path, sizeof(tmpfile_path), "%s%s",
- node->name, XB_TMPFILE_SUFFIX);
-
- tmpfile = ds_open(ds_buffer, tmpfile_path, &cursor.statinfo);
- if (tmpfile == NULL) {
-
- msg("Could not open temporary file '%s'\n", tmpfile_path);
- goto error;
- }
-
- have_next_range = page_map_file_next(pmap_cur, &pmap_range);
-
- page_expected_no = 0;
-
- /* Initialize and mark the empty page which is used to replace
- skipped pages. */
- memset(empty_page, 0, cursor.page_size);
- memcpy(empty_page + compacted_page_magic_offset,
- compacted_page_magic, compacted_page_magic_size);
- mach_write_to_4(empty_page + FIL_PAGE_SPACE_OR_CHKSUM,
- BUF_NO_CHECKSUM_MAGIC);
- mach_write_to_4(empty_page + cursor.page_size -
- FIL_PAGE_END_LSN_OLD_CHKSUM,
- BUF_NO_CHECKSUM_MAGIC);
-
-
- /* Main copy loop */
-
- while ((res = xb_fil_cur_read(&cursor)) == XB_FIL_CUR_SUCCESS) {
-
- for (i = 0, page = cursor.buf; i < cursor.buf_npages;
- i++, page += cursor.page_size) {
-
- ulint page_read_no;
-
- page_read_no = mach_read_from_4(page + FIL_PAGE_OFFSET);
- xb_a(!page_read_no || page_expected_no <= page_read_no);
-
- if (have_next_range &&
- page_expected_no == pmap_range.from) {
-
- xb_a(pmap_range.from <= pmap_range.to);
-
- /* Write empty pages instead of skipped ones, if
- necessary. */
-
- while (page_expected_no <= pmap_range.to) {
-
- if (ds_write(tmpfile, empty_page,
- cursor.page_size)) {
-
- goto write_error;
- }
-
- page_expected_no++;
- }
-
- have_next_range =
- page_map_file_next(pmap_cur,
- &pmap_range);
- }
-
- /* Write the current page */
-
- if (ds_write(tmpfile, page, cursor.page_size)) {
-
- goto write_error;
- }
-
- page_expected_no++;
- }
- }
-
- if (res != XB_FIL_CUR_EOF) {
-
- goto error;
- }
-
- /* Write empty pages instead of trailing skipped ones, if any */
-
- if (have_next_range) {
-
- xb_a(page_expected_no == pmap_range.from);
- xb_a(pmap_range.from <= pmap_range.to);
-
- while (page_expected_no <= pmap_range.to) {
-
- if (ds_write(tmpfile, empty_page,
- cursor.page_size)) {
-
- goto write_error;
- }
-
- page_expected_no++;
- }
-
- xb_a(!page_map_file_next(pmap_cur, &pmap_range));
- }
-
- /* Replace the original .ibd file with the expanded file */
- if (my_rename(tmpfile_path, node->name, MYF(MY_WME))) {
-
- msg("Failed to rename '%s' to '%s'\n",
- tmpfile_path, node->name);
- goto error;
- }
-
- my_delete(pmapfile_path, MYF(MY_WME));
-
- if (!ds_close(tmpfile)) {
- success = TRUE;
- }
- tmpfile = NULL;
-
- goto end;
-
-write_error:
- msg("Write to '%s' failed\n", tmpfile_path);
-
-error:
- if (tmpfile != NULL) {
-
- ds_close(tmpfile);
- my_delete(tmpfile_path, MYF(MY_WME));
- }
-
-end:
- ds_destroy(ds_buffer);
- ds_destroy(ds_local);
-
- xb_fil_cur_close(&cursor);
-
- page_map_file_close(pmap_cur);
-
- return(success);
-}
-
-/******************************************************************************
-Expand the data files according to the skipped pages maps created by --compact.
-@return TRUE on success, FALSE on failure. */
-my_bool
-xb_expand_datafiles(void)
-/*=====================*/
-{
- ulint nfiles;
- datafiles_iter_t *it = NULL;
- fil_node_t *node;
- fil_space_t *space;
-
- msg("Starting to expand compacted .ibd files.\n");
-
- /* Initialize the tablespace cache */
- if (xb_data_files_init() != DB_SUCCESS) {
- return(FALSE);
- }
-
- nfiles = UT_LIST_GET_LEN(fil_system->space_list);
- xb_a(nfiles > 0);
-
- it = datafiles_iter_new(fil_system);
- if (it == NULL) {
- msg("xtrabackup: error: datafiles_iter_new() failed.\n");
- goto error;
- }
-
- while ((node = datafiles_iter_next(it)) != NULL) {
-
- space = node->space;
-
- /* System tablespace cannot be compacted */
- if (!fil_is_user_tablespace_id(space->id)) {
-
- continue;
- }
-
- if (!xb_expand_file(node)) {
-
- goto error;
- }
- }
-
- datafiles_iter_free(it);
- xb_data_files_close();
-
- return(TRUE);
-
-error:
- if (it != NULL) {
- datafiles_iter_free(it);
- }
-
- xb_data_files_close();
-
- return(FALSE);
-}
-
-/******************************************************************************
-Callback used in buf_page_io_complete() to detect compacted pages.
-@return TRUE if the page is marked as compacted, FALSE otherwise. */
-ibool
-buf_page_is_compacted(
-/*==================*/
- const byte* page) /*!< in: a database page */
-{
- return !memcmp(page + compacted_page_magic_offset,
- compacted_page_magic, compacted_page_magic_size);
-}
-
-/*****************************************************************************
-Builds an index definition corresponding to an index object. It is roughly
-similar to innobase_create_index_def() / innobase_create_index_field_def() and
-the opposite to dict_mem_index_create() / dict_mem_index_add_field(). */
-static
-void
-xb_build_index_def(
-/*=======================*/
- mem_heap_t* heap, /*!< in: heap */
- const dict_index_t* index, /*!< in: index */
- index_def_t* index_def) /*!< out: index definition */
-{
- index_field_t* fields;
- ulint n_fields;
- ulint i;
-
- ut_a(index->n_fields);
- ut_ad(index->magic_n == DICT_INDEX_MAGIC_N);
-
- /* Use n_user_defined_cols instead of n_fields, as the index will
- contain a part of the primary key after n_user_defined_cols, and those
- columns will be created automatically in
- dict_index_build_internal_clust(). */
- n_fields = index->n_user_defined_cols;
-
- memset(index_def, 0, sizeof(*index_def));
-
- index_def->name = mem_heap_strdup(heap, index->name);
- index_def->ind_type = index->type;
-
- fields = static_cast<index_field_t *>
- (mem_heap_alloc(heap, n_fields * sizeof(*fields)));
-
- for (i = 0; i < n_fields; i++) {
- dict_field_t* field;
-
- field = dict_index_get_nth_field(index, i);
- fields[i].col_no = dict_col_get_no(field->col);
- fields[i].prefix_len = field->prefix_len;
- }
-
- index_def->fields = fields;
- index_def->n_fields = n_fields;
-}
-
-/* A dummy autoc_inc sequence for row_merge_build_indexes(). */
-static ib_sequence_t null_seq(NULL, 0, 0);
-/* A dummy table share and table for row_merge_build_indexes() error reporting.
-Assumes that no errors are going to be reported. */
-static struct TABLE_SHARE dummy_table_share;
-static struct TABLE dummy_table;
-
-/********************************************************************//**
-Rebuild secondary indexes for a given table. */
-static
-void
-xb_rebuild_indexes_for_table(
-/*=========================*/
- dict_table_t* table, /*!< in: table */
- trx_t* trx, /*!< in: transaction handle */
- ulint thread_n) /*!< in: thread number */
-{
- dict_index_t* index;
- dict_index_t** indexes;
- ulint n_indexes;
- index_def_t* index_defs;
- ulint i;
- mem_heap_t* heap;
- ulint error;
- ulint* add_key_nums;
-
- ut_ad(!mutex_own(&(dict_sys->mutex)));
- ut_ad(table);
-
- ut_a(UT_LIST_GET_LEN(table->indexes) > 0);
-
- n_indexes = UT_LIST_GET_LEN(table->indexes) - 1;
- if (!n_indexes) {
- /* Only the primary key, nothing to do. */
- return;
- }
-
- heap = mem_heap_create(1024);
-
- indexes = (dict_index_t**) mem_heap_alloc(heap,
- n_indexes * sizeof(*indexes));
- index_defs = (index_def_t*) mem_heap_alloc(heap, n_indexes *
- sizeof(*index_defs));
- add_key_nums = static_cast<ulint *>
- (mem_heap_alloc(heap, n_indexes * sizeof(*add_key_nums)));
-
- /* Skip the primary key. */
- index = dict_table_get_first_index(table);
- ut_a(dict_index_is_clust(index));
-
- row_mysql_lock_data_dictionary(trx);
-
- for (i = 0; (index = dict_table_get_next_index(index)); i++) {
-
- msg("[%02lu] Found index %s\n", thread_n, index->name);
-
- /* Pretend that it's the current trx that created this index.
- Required to avoid 5.6+ debug assertions. */
- index->trx_id = trx->id;
-
- xb_build_index_def(heap, index, &index_defs[i]);
-
- /* In 5.6+, row_merge_drop_indexes() drops all the indexes on
- the table that have the temp index prefix. It does not accept
- an array of indexes to drop as in 5.5-. */
- row_merge_rename_index_to_drop(trx, table->id, index->id);
- }
-
- ut_ad(i == n_indexes);
-
- row_merge_drop_indexes(trx, table, TRUE);
-
- index = dict_table_get_first_index(table);
- ut_a(dict_index_is_clust(index));
- index = dict_table_get_next_index(index);
- while (index) {
-
- /* In 5.6+, row_merge_drop_indexes() does not remove the
- indexes from the dictionary cache nor from any foreign key
- list. This may cause invalid dereferences as we try to access
- the dropped indexes from other tables as FKs. */
-
- dict_index_t* next_index = dict_table_get_next_index(index);
- index->to_be_dropped = 1;
-
- /* Patch up any FK referencing this index with NULL */
- dict_foreign_replace_index(table, NULL, index);
-
- dict_index_remove_from_cache(table, index);
-
- index = next_index;
- }
-
- msg("[%02lu] Rebuilding %lu index(es).\n", thread_n, n_indexes);
-
- error = row_merge_lock_table(trx, table, LOCK_X);
- xb_a(error == DB_SUCCESS);
-
- for (i = 0; i < n_indexes; i++) {
- indexes[i] = row_merge_create_index(trx, table,
- &index_defs[i]);
- add_key_nums[i] = index_defs[i].key_number;
- }
-
- /* Commit trx to release latches on system tables */
- trx_commit_for_mysql(trx);
- trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
-
- row_mysql_unlock_data_dictionary(trx);
-
- /* Reacquire table lock for row_merge_build_indexes() */
- error = row_merge_lock_table(trx, table, LOCK_X);
- xb_a(error == DB_SUCCESS);
-
- error = row_merge_build_indexes(trx, table, table, FALSE, indexes,
- add_key_nums, n_indexes, &dummy_table,
- NULL, NULL, ULINT_UNDEFINED, null_seq);
- ut_a(error == DB_SUCCESS);
-
- mem_heap_free(heap);
-
- trx_commit_for_mysql(trx);
-
- trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
-}
-
-/**************************************************************************
-Worker thread function for index rebuild. */
-static
-void *
-xb_rebuild_indexes_thread_func(
-/*===========================*/
- void* arg) /* thread context */
-{
- dict_table_t* table;
- index_rebuild_table_t* rebuild_table;
- index_rebuild_thread_t* thread;
- trx_t* trx;
-
- thread = (index_rebuild_thread_t *) arg;
-
- trx = trx_allocate_for_mysql();
-
- /* Suppress foreign key checks, as we are going to drop and recreate all
- secondary keys. */
- trx->check_foreigns = FALSE;
- trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
-
- /* Loop until there are no more tables in tables list */
- for (;;) {
- pthread_mutex_lock(&table_list_mutex);
-
- rebuild_table = UT_LIST_GET_FIRST(table_list);
-
- if (rebuild_table == NULL) {
-
- pthread_mutex_unlock(&table_list_mutex);
- break;
- }
-
- UT_LIST_REMOVE(list, table_list, rebuild_table);
-
- pthread_mutex_unlock(&table_list_mutex);
-
- ut_ad(rebuild_table->name);
- ut_ad(fil_is_user_tablespace_id(rebuild_table->space_id));
-
- row_mysql_lock_data_dictionary(trx);
-
- table = dict_table_get_low(rebuild_table->name);
-
- ut_d(table->n_ref_count++);
-
- row_mysql_unlock_data_dictionary(trx);
-
- ut_a(table != NULL);
- ut_a(table->space == rebuild_table->space_id);
-
- /* Discard change buffer entries for this space */
- ibuf_delete_for_discarded_space(rebuild_table->space_id);
-
- msg("[%02lu] Checking if there are indexes to rebuild in table "
- "%s (space id: %lu)\n",
- thread->num,
- rebuild_table->name, rebuild_table->space_id);
-
- xb_rebuild_indexes_for_table(table, trx, thread->num);
-
- ut_d(table->n_ref_count--);
-
- mem_free(rebuild_table->name);
- mem_free(rebuild_table);
- }
-
- trx_commit_for_mysql(trx);
-
- trx_free_for_mysql(trx);
-
- return(NULL);
-}
-
-/******************************************************************************
-Rebuild all secondary indexes in all tables in separate spaces. Called from
-innobase_start_or_create_for_mysql(). */
-void
-xb_compact_rebuild_indexes(void)
-/*=============================*/
-{
- dict_table_t* sys_tables;
- dict_index_t* sys_index;
- btr_pcur_t pcur;
- const rec_t* rec;
- mtr_t mtr;
- const byte* field;
- ulint len;
- ulint space_id;
- trx_t* trx;
- index_rebuild_table_t* rebuild_table;
- index_rebuild_thread_t* threads;
- ulint i;
-
- /* Set up the dummy table for the index rebuild error reporting */
- dummy_table_share.fields = 0;
- dummy_table.s = &dummy_table_share;
-
- /* Iterate all tables that are not in the system tablespace and add them
- to the list of tables to be rebuilt later. */
-
- trx = trx_allocate_for_mysql();
- trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
-
- row_mysql_lock_data_dictionary(trx);
-
- /* Enlarge the fatal lock wait timeout during index rebuild
- operation. */
- os_increment_counter_by_amount(server_mutex,
- srv_fatal_semaphore_wait_threshold,
- 7200);
-
- mtr_start(&mtr);
-
- sys_tables = dict_table_get_low("SYS_TABLES");
- sys_index = UT_LIST_GET_FIRST(sys_tables->indexes);
- ut_a(!dict_table_is_comp(sys_tables));
-
- pthread_mutex_init(&table_list_mutex, NULL);
- UT_LIST_INIT(table_list);
-
- btr_pcur_open_at_index_side(TRUE, sys_index, BTR_SEARCH_LEAF, &pcur,
- TRUE, 0, &mtr);
- for (;;) {
- btr_pcur_move_to_next_user_rec(&pcur, &mtr);
-
- rec = btr_pcur_get_rec(&pcur);
-
- if (!btr_pcur_is_on_user_rec(&pcur)) {
- /* end of index */
-
- break;
- }
-
- if (rec_get_deleted_flag(rec, 0)) {
- continue;
- }
-
- field = rec_get_nth_field_old(rec, 9, &len);
- ut_a(len == 4);
-
- space_id = mach_read_from_4(field);
-
- /* Don't touch tables in the system tablespace */
- if (!fil_is_user_tablespace_id(space_id)) {
-
- continue;
- }
-
- field = rec_get_nth_field_old(rec, 0, &len);
-
- rebuild_table = static_cast<index_rebuild_table_t *>
- (mem_alloc(sizeof(*rebuild_table)));
- rebuild_table->name = mem_strdupl((char*) field, len);
- rebuild_table->space_id = space_id;
-
- UT_LIST_ADD_LAST(list, table_list, rebuild_table);
- }
-
- btr_pcur_close(&pcur);
- mtr_commit(&mtr);
-
- row_mysql_unlock_data_dictionary(trx);
-
- trx_commit_for_mysql(trx);
-
- trx_free_for_mysql(trx);
-
- /* Start worker threads for the index rebuild operation */
- ut_ad(xtrabackup_rebuild_threads > 0);
-
- if (xtrabackup_rebuild_threads > 1) {
- msg("Starting %lu threads to rebuild indexes.\n",
- xtrabackup_rebuild_threads);
- }
-
- threads = (index_rebuild_thread_t *)
- mem_alloc(sizeof(*threads) *
- xtrabackup_rebuild_threads);
-
- for (i = 0; i < xtrabackup_rebuild_threads; i++) {
-
- threads[i].num = i+1;
- if (pthread_create(&threads[i].id, NULL,
- xb_rebuild_indexes_thread_func,
- &threads[i])) {
-
- msg("error: pthread_create() failed: errno = %d\n",
- errno);
- ut_a(0);
- }
- }
-
- /* Wait for worker threads to finish */
- for (i = 0; i < xtrabackup_rebuild_threads; i++) {
- pthread_join(threads[i].id, NULL);
- }
-
- mem_free(threads);
-}
diff --git a/extra/mariabackup/compact.h b/extra/mariabackup/compact.h
deleted file mode 100644
index d0d9840f66d..00000000000
--- a/extra/mariabackup/compact.h
+++ /dev/null
@@ -1,44 +0,0 @@
-/******************************************************
-XtraBackup: hot backup tool for InnoDB
-(c) 2009-2013 Percona LLC and/or its affiliates.
-Originally Created 3/3/2009 Yasufumi Kinoshita
-Written by Alexey Kopytov, Aleksandr Kuzminsky, Stewart Smith, Vadim Tkachenko,
-Yasufumi Kinoshita, Ignacio Nin and Baron Schwartz.
-
-This program is free software; you can redistribute it and/or modify
-it under the terms of the GNU General Public License as published by
-the Free Software Foundation; version 2 of the License.
-
-This program is distributed in the hope that it will be useful,
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU General Public License for more details.
-
-You should have received a copy of the GNU General Public License
-along with this program; if not, write to the Free Software
-Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
-
-*******************************************************/
-
-#ifndef XB_COMPACT_H
-#define XB_COMPACT_H
-
-#include "write_filt.h"
-
-/* Compact page filter context */
-typedef struct {
- my_bool skip;
- ds_ctxt_t *ds_buffer;
- ds_file_t *buffer;
- index_id_t clustered_index;
- my_bool clustered_index_found;
- my_bool inside_skipped_range;
- ulint free_limit;
-} xb_wf_compact_ctxt_t;
-
-/******************************************************************************
-Expand the data files according to the skipped pages maps created by --compact.
-@return TRUE on success, FALSE on failure. */
-my_bool xb_expand_datafiles(void);
-
-#endif
diff --git a/extra/mariabackup/datasink.c b/extra/mariabackup/datasink.c
index 2f4233ddc98..199eb77ad55 100644
--- a/extra/mariabackup/datasink.c
+++ b/extra/mariabackup/datasink.c
@@ -46,7 +46,12 @@ ds_create(const char *root, ds_type_t type)
ds = &datasink_local;
break;
case DS_TYPE_ARCHIVE:
+#ifdef HAVE_LIBARCHIVE
ds = &datasink_archive;
+#else
+ msg("Error : mariabackup was built without libarchive support");
+ exit(EXIT_FAILURE);
+#endif
break;
case DS_TYPE_XBSTREAM:
ds = &datasink_xbstream;
@@ -55,8 +60,10 @@ ds_create(const char *root, ds_type_t type)
ds = &datasink_compress;
break;
case DS_TYPE_ENCRYPT:
- ds = &datasink_encrypt;
+ msg("Error : mariabackup does not support encrypted backups.");
+ exit(EXIT_FAILURE);
break;
+
case DS_TYPE_TMPFILE:
ds = &datasink_tmpfile;
break;
diff --git a/extra/mariabackup/datasink.h b/extra/mariabackup/datasink.h
index 36a3d564a57..e378a806441 100644
--- a/extra/mariabackup/datasink.h
+++ b/extra/mariabackup/datasink.h
@@ -28,6 +28,7 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
extern "C" {
#endif
+extern char *xtrabackup_tmpdir;
struct datasink_struct;
typedef struct datasink_struct datasink_t;
diff --git a/extra/mariabackup/ds_compress.c b/extra/mariabackup/ds_compress.c
index c49fceaf7fa..15801c8abd4 100644
--- a/extra/mariabackup/ds_compress.c
+++ b/extra/mariabackup/ds_compress.c
@@ -152,7 +152,7 @@ compress_open(ds_ctxt_t *ctxt, const char *path, MY_STAT *mystat)
/* Write the qpress file header */
name_len = strlen(new_name);
if (ds_write(dest_file, "F", 1) ||
- write_uint32_le(dest_file, name_len) ||
+ write_uint32_le(dest_file, (uint)name_len) ||
/* we want to write the terminating \0 as well */
ds_write(dest_file, new_name, name_len + 1)) {
goto err;
@@ -453,7 +453,7 @@ compress_worker_thread_func(void *arg)
with qpress implementation. */
thd->adler = adler32(0x00000001, (uchar *) thd->to,
- thd->to_len);
+ (uInt)thd->to_len);
}
pthread_mutex_unlock(&thd->data_mutex);
diff --git a/extra/mariabackup/ds_encrypt.c b/extra/mariabackup/ds_encrypt.c
index f8d62a03e13..19ced416687 100644
--- a/extra/mariabackup/ds_encrypt.c
+++ b/extra/mariabackup/ds_encrypt.c
@@ -22,7 +22,7 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
#include <my_base.h>
#include "common.h"
#include "datasink.h"
-
+#ifdef HAVE_GRYPT
#if GCC_VERSION >= 4002
/* Workaround to avoid "gcry_ac_* is deprecated" warnings in gcrypt.h */
# pragma GCC diagnostic ignored "-Wdeprecated-declarations"
@@ -615,3 +615,4 @@ encrypt_worker_thread_func(void *arg)
return NULL;
}
+#endif /* HAVE_GCRYPT*/
diff --git a/extra/mariabackup/ds_encrypt.h b/extra/mariabackup/ds_encrypt.h
index ed869747d79..3e9e40ad354 100644
--- a/extra/mariabackup/ds_encrypt.h
+++ b/extra/mariabackup/ds_encrypt.h
@@ -22,7 +22,7 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
#define DS_ENCRYPT_H
#include "datasink.h"
-
+#ifdef HAVE_GCRYPT
extern datasink_t datasink_encrypt;
-
+#endif
#endif
diff --git a/extra/mariabackup/ds_local.c b/extra/mariabackup/ds_local.c
index dc13ed7595e..3e2b1e0129b 100644
--- a/extra/mariabackup/ds_local.c
+++ b/extra/mariabackup/ds_local.c
@@ -53,9 +53,9 @@ local_init(const char *root)
&& my_errno != EEXIST && my_errno != EISDIR)
{
char errbuf[MYSYS_STRERROR_SIZE];
+ my_strerror(errbuf, sizeof(errbuf),my_errno);
my_error(EE_CANT_MKDIR, MYF(ME_BELL | ME_WAITTANG),
- root, my_errno, my_strerror(errbuf, sizeof(errbuf),
- my_errno));
+ root, my_errno,errbuf, my_errno);
return NULL;
}
@@ -85,9 +85,9 @@ local_open(ds_ctxt_t *ctxt, const char *path,
dirname_part(dirpath, fullpath, &dirpath_len);
if (my_mkdir(dirpath, 0777, MYF(0)) < 0 && my_errno != EEXIST) {
char errbuf[MYSYS_STRERROR_SIZE];
+ my_strerror(errbuf, sizeof(errbuf), my_errno);
my_error(EE_CANT_MKDIR, MYF(ME_BELL | ME_WAITTANG),
- dirpath, my_errno, my_strerror(errbuf, sizeof(errbuf),
- my_errno));
+ dirpath, my_errno, errbuf);
return NULL;
}
diff --git a/extra/mariabackup/ds_stdout.c b/extra/mariabackup/ds_stdout.c
index 616bcbd831e..91a514ddf64 100644
--- a/extra/mariabackup/ds_stdout.c
+++ b/extra/mariabackup/ds_stdout.c
@@ -79,7 +79,7 @@ stdout_open(ds_ctxt_t *ctxt __attribute__((unused)),
setmode(fileno(stdout), _O_BINARY);
#endif
- stdout_file->fd = fileno(stdout);
+ stdout_file->fd = my_fileno(stdout);
file->path = (char *) stdout_file + sizeof(ds_stdout_file_t);
memcpy(file->path, fullpath, pathlen);
diff --git a/extra/mariabackup/ds_tmpfile.c b/extra/mariabackup/ds_tmpfile.c
index 915191dcdae..b039d83ba03 100644
--- a/extra/mariabackup/ds_tmpfile.c
+++ b/extra/mariabackup/ds_tmpfile.c
@@ -53,7 +53,6 @@ datasink_t datasink_tmpfile = {
&tmpfile_deinit
};
-MY_TMPDIR mysql_tmpdir_list;
static ds_ctxt_t *
tmpfile_init(const char *root)
@@ -90,7 +89,7 @@ tmpfile_open(ds_ctxt_t *ctxt, const char *path,
/* Create a temporary file in tmpdir. The file will be automatically
removed on close. Code copied from mysql_tmpfile(). */
- fd = create_temp_file(tmp_path, my_tmpdir(&mysql_tmpdir_list),
+ fd = create_temp_file(tmp_path,xtrabackup_tmpdir,
"xbtemp",
#ifdef __WIN__
O_BINARY | O_TRUNC | O_SEQUENTIAL |
diff --git a/extra/mariabackup/encryption_plugin.cc b/extra/mariabackup/encryption_plugin.cc
new file mode 100644
index 00000000000..9f2782d89a1
--- /dev/null
+++ b/extra/mariabackup/encryption_plugin.cc
@@ -0,0 +1,157 @@
+#include <mysqld.h>
+#include <mysql.h>
+#include <xtrabackup.h>
+#include <encryption_plugin.h>
+#include <backup_copy.h>
+#include <sql_plugin.h>
+#include <sstream>
+#include <vector>
+#include <common.h>
+#include <backup_mysql.h>
+
+
+extern struct st_maria_plugin *mysql_optional_plugins[];
+extern struct st_maria_plugin *mysql_mandatory_plugins[];
+static void encryption_plugin_init(int argc, char **argv);
+
+extern char *xb_plugin_load;
+extern char *xb_plugin_dir;
+
+const int PLUGIN_MAX_ARGS = 1024;
+vector<string> backup_plugins_args;
+
+const char *QUERY_PLUGIN =
+"SELECT plugin_name, plugin_library, @@plugin_dir"
+" FROM information_schema.plugins WHERE plugin_type='ENCRYPTION'"
+" AND plugin_status='ACTIVE'";
+
+string encryption_plugin_config;
+
+static void add_to_plugin_load_list(const char *plugin_def)
+{
+ opt_plugin_load_list_ptr->push_back(new i_string(plugin_def));
+}
+
+static char XTRABACKUP_EXE[] = "xtrabackup";
+
+void encryption_plugin_backup_init(MYSQL *mysql)
+{
+ MYSQL_RES *result;
+ MYSQL_ROW row;
+ ostringstream oss;
+ char *argv[PLUGIN_MAX_ARGS];
+ int argc;
+
+ result = xb_mysql_query(mysql, QUERY_PLUGIN, true, true);
+ row = mysql_fetch_row(result);
+ if (!row)
+ {
+ mysql_free_result(result);
+ return;
+ }
+
+ char *name= row[0];
+ char *library= row[1];
+ char *dir= row[2];
+
+#ifdef _WIN32
+ for (char *p = dir; *p; p++)
+ if (*p == '\\') *p = '/';
+#endif
+
+ string plugin_load(name);
+ if (library)
+ plugin_load += string("=") + library;
+
+ oss << "plugin_load=" << plugin_load << endl;
+
+ /* Required to load the plugin later.*/
+ add_to_plugin_load_list(plugin_load.c_str());
+ strncpy(opt_plugin_dir, dir, FN_REFLEN);
+
+ oss << "plugin_dir=" << '"' << dir << '"' << endl;
+
+
+ /* Read plugin variables. */
+ char query[1024];
+ snprintf(query, 1024, "SHOW variables like '%s_%%'", name);
+ mysql_free_result(result);
+
+ result = xb_mysql_query(mysql, query, true, true);
+ while ((row = mysql_fetch_row(result)))
+ {
+ string arg("--");
+ arg += row[0];
+ arg += "=";
+ arg += row[1];
+ backup_plugins_args.push_back(arg);
+ oss << row[0] << "=" << row[1] << endl;
+ }
+
+ mysql_free_result(result);
+
+ /* Check whether to encrypt logs. */
+ result = xb_mysql_query(mysql, "select @@innodb_encrypt_log", true, true);
+ row = mysql_fetch_row(result);
+ srv_encrypt_log = (row != 0 && row[0][0] == '1');
+ oss << "innodb_encrypt_log=" << row[0] << endl;
+
+ mysql_free_result(result);
+
+ encryption_plugin_config = oss.str();
+
+ argc = 0;
+ argv[argc++] = XTRABACKUP_EXE;
+ for(size_t i = 0; i < backup_plugins_args.size(); i++)
+ {
+ argv[argc++] = (char *)backup_plugins_args[i].c_str();
+ if (argc == PLUGIN_MAX_ARGS - 2)
+ break;
+ }
+ argv[argc] = 0;
+
+ encryption_plugin_init(argc, argv);
+}
+
+const char *encryption_plugin_get_config()
+{
+ return encryption_plugin_config.c_str();
+}
+
+extern int finalize_encryption_plugin(st_plugin_int *plugin);
+
+
+void encryption_plugin_prepare_init(int argc, char **argv)
+{
+
+ if (!xb_plugin_load)
+ {
+ /* This prevents crashes e.g in --stats with wrong my.cnf*/
+ finalize_encryption_plugin(0);
+ return;
+ }
+
+ add_to_plugin_load_list(xb_plugin_load);
+
+ if (xb_plugin_dir)
+ strncpy(opt_plugin_dir, xb_plugin_dir, FN_REFLEN);
+
+ char **new_argv = new char *[argc + 1];
+ new_argv[0] = XTRABACKUP_EXE;
+ memcpy(&new_argv[1], argv, argc*sizeof(char *));
+
+ encryption_plugin_init(argc+1, new_argv);
+
+ delete[] new_argv;
+}
+
+static void encryption_plugin_init(int argc, char **argv)
+{
+ /* Patch optional and mandatory plugins, we only need to load the one in xb_plugin_load. */
+ mysql_optional_plugins[0] = mysql_mandatory_plugins[0] = 0;
+ msg("Loading encryption plugin\n");
+ for (int i= 1; i < argc; i++)
+ msg("\t Encryption plugin parameter : '%s'\n", argv[i]);
+ plugin_init(&argc, argv, PLUGIN_INIT_SKIP_PLUGIN_TABLE);
+}
+
diff --git a/extra/mariabackup/encryption_plugin.h b/extra/mariabackup/encryption_plugin.h
new file mode 100644
index 00000000000..16d74790254
--- /dev/null
+++ b/extra/mariabackup/encryption_plugin.h
@@ -0,0 +1,7 @@
+#include <mysql.h>
+#include <string>
+extern void encryption_plugin_backup_init(MYSQL *mysql);
+extern const char* encryption_plugin_get_config();
+extern void encryption_plugin_prepare_init(int argc, char **argv);
+
+//extern void encryption_plugin_init(int argc, char **argv);
diff --git a/extra/mariabackup/fil_cur.cc b/extra/mariabackup/fil_cur.cc
index 22ebdb90215..820d8e10c29 100644
--- a/extra/mariabackup/fil_cur.cc
+++ b/extra/mariabackup/fil_cur.cc
@@ -33,6 +33,7 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
#include "common.h"
#include "read_filt.h"
#include "xtrabackup.h"
+#include "xb0xb.h"
/* Size of read buffer in pages (640 pages = 10M for 16K sized pages) */
#define XB_FIL_CUR_PAGES 640
@@ -167,7 +168,7 @@ xb_fil_cur_open(
os_file_create_simple_no_error_handling(0, node->name,
OS_FILE_OPEN,
OS_FILE_READ_ONLY,
- &success);
+ &success,0);
if (!success) {
/* The following call prints an error message */
os_file_get_last_error(TRUE);
@@ -200,7 +201,7 @@ xb_fil_cur_open(
cursor->node = node;
cursor->file = node->handle;
- if (my_fstat(cursor->file, &cursor->statinfo, MYF(MY_WME))) {
+ if (stat(cursor->abs_path, &cursor->statinfo)) {
msg("[%02u] xtrabackup: error: cannot stat %s\n",
thread_n, cursor->abs_path);
@@ -253,7 +254,7 @@ xb_fil_cur_open(
cursor->buf_page_no = 0;
cursor->thread_n = thread_n;
- cursor->space_size = cursor->statinfo.st_size / page_size;
+ cursor->space_size = (ulint)(cursor->statinfo.st_size / page_size);
cursor->read_filter = read_filter;
cursor->read_filter->init(&cursor->read_filter_ctxt, cursor,
@@ -327,26 +328,32 @@ read_retry:
cursor->buf_read = 0;
cursor->buf_npages = 0;
cursor->buf_offset = offset;
- cursor->buf_page_no = (ulint) (offset >> cursor->page_size_shift);
+ cursor->buf_page_no = (ulint)(offset >> cursor->page_size_shift);
success = os_file_read(cursor->file, cursor->buf, offset,
- to_read);
+ (ulint)to_read);
if (!success) {
return(XB_FIL_CUR_ERROR);
}
+ fil_system_enter();
+ fil_space_t *space = fil_space_get_by_id(cursor->space_id);
+ fil_system_exit();
+
/* check pages for corruption and re-read if necessary. i.e. in case of
partially written pages */
for (page = cursor->buf, i = 0; i < npages;
page += cursor->page_size, i++) {
+ ib_int64_t page_no = cursor->buf_page_no + i;
- if (buf_page_is_corrupted(TRUE, page, cursor->zip_size)) {
+ bool checksum_ok = fil_space_verify_crypt_checksum(page, cursor->zip_size,space, (ulint)page_no);
- ulint page_no = cursor->buf_page_no + i;
+ if (!checksum_ok &&
+ buf_page_is_corrupted(true, page, cursor->zip_size,space)) {
if (cursor->is_system &&
- page_no >= FSP_EXTENT_SIZE &&
- page_no < FSP_EXTENT_SIZE * 3) {
+ page_no >= (ib_int64_t)FSP_EXTENT_SIZE &&
+ page_no < (ib_int64_t) FSP_EXTENT_SIZE * 3) {
/* skip doublewrite buffer pages */
xb_a(cursor->page_size == UNIV_PAGE_SIZE);
msg("[%02u] xtrabackup: "
diff --git a/extra/mariabackup/fil_cur.h b/extra/mariabackup/fil_cur.h
index 2057765dab5..88239efd2bb 100644
--- a/extra/mariabackup/fil_cur.h
+++ b/extra/mariabackup/fil_cur.h
@@ -49,10 +49,10 @@ struct xb_fil_cur_t {
/*!< read filter context */
byte* orig_buf; /*!< read buffer */
byte* buf; /*!< aligned pointer for orig_buf */
- ulint buf_size; /*!< buffer size in bytes */
- ulint buf_read; /*!< number of read bytes in buffer
+ size_t buf_size; /*!< buffer size in bytes */
+ size_t buf_read; /*!< number of read bytes in buffer
after the last cursor read */
- ulint buf_npages; /*!< number of pages in buffer after the
+ size_t buf_npages; /*!< number of pages in buffer after the
last cursor read */
ib_int64_t buf_offset; /*!< file offset of the first page in
buffer */
diff --git a/extra/mariabackup/innobackupex.cc b/extra/mariabackup/innobackupex.cc
index ec697bb56ff..59fb8fb5565 100644
--- a/extra/mariabackup/innobackupex.cc
+++ b/extra/mariabackup/innobackupex.cc
@@ -52,14 +52,12 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#include <mysqld.h>
#include <my_default.h>
#include <my_getopt.h>
-#include <strings.h>
#include <string>
#include <sstream>
#include <set>
#include "common.h"
#include "innobackupex.h"
#include "xtrabackup.h"
-#include "xtrabackup_version.h"
#include "xbstream.h"
#include "fil_cur.h"
#include "write_filt.h"
@@ -667,22 +665,6 @@ static struct my_option ibx_long_options[] =
(uchar*) &ibx_xtrabackup_parallel, (uchar*) &ibx_xtrabackup_parallel,
0, GET_INT, REQUIRED_ARG, 1, 1, INT_MAX, 0, 0, 0},
- {"rebuild-indexes", OPT_REBUILD_INDEXES,
- "This option only has effect when used together with the --apply-log "
- "option and is passed directly to xtrabackup. When used, makes "
- "xtrabackup rebuild all secondary indexes after applying the log. "
- "This option is normally used to prepare compact backups. See the "
- "XtraBackup manual for more information.",
- (uchar*) &ibx_xtrabackup_rebuild_indexes,
- (uchar*) &ibx_xtrabackup_rebuild_indexes,
- 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
-
- {"rebuild-threads", OPT_REBUILD_THREADS,
- "Use this number of threads to rebuild indexes in a compact backup. "
- "Only has effect with --prepare and --rebuild-indexes.",
- (uchar*) &ibx_xtrabackup_rebuild_threads,
- (uchar*) &ibx_xtrabackup_rebuild_threads,
- 0, GET_UINT, REQUIRED_ARG, 1, 1, UINT_MAX, 0, 0, 0},
{"stream", OPT_STREAM, "This option specifies the format in which to "
"do the streamed backup. The option accepts a string argument. The "
@@ -845,9 +827,9 @@ ibx_get_one_option(int optid,
exit(0);
break;
case 'v':
- msg("innobackupex version %s %s (%s) (revision id: %s)\n",
- XTRABACKUP_VERSION,
- SYSTEM_TYPE, MACHINE_TYPE, XTRABACKUP_REVISION);
+ msg("innobackupex version %s %s (%s)\n",
+ MYSQL_SERVER_VERSION,
+ SYSTEM_TYPE, MACHINE_TYPE);
exit(0);
break;
case OPT_HISTORY:
@@ -1039,7 +1021,6 @@ ibx_init()
/* setup xtrabackup options */
xb_close_files = ibx_xb_close_files;
- xtrabackup_compact = ibx_xtrabackup_compact;
xtrabackup_compress_alg = ibx_xtrabackup_compress_alg;
xtrabackup_compress_threads = ibx_xtrabackup_compress_threads;
xtrabackup_compress_chunk_size = ibx_xtrabackup_compress_chunk_size;
@@ -1057,8 +1038,6 @@ ibx_init()
xtrabackup_log_copy_interval = ibx_xtrabackup_log_copy_interval;
xtrabackup_incremental = ibx_xtrabackup_incremental;
xtrabackup_parallel = ibx_xtrabackup_parallel;
- xtrabackup_rebuild_indexes = ibx_xtrabackup_rebuild_indexes;
- xtrabackup_rebuild_threads = ibx_xtrabackup_rebuild_threads;
xtrabackup_stream_str = ibx_xtrabackup_stream_str;
xtrabackup_tables_file = ibx_xtrabackup_tables_file;
xtrabackup_throttle = ibx_xtrabackup_throttle;
diff --git a/extra/mariabackup/read_filt.cc b/extra/mariabackup/read_filt.cc
index 8ebc735e99e..05e6b7c86c7 100644
--- a/extra/mariabackup/read_filt.cc
+++ b/extra/mariabackup/read_filt.cc
@@ -75,7 +75,7 @@ rf_pass_through_get_next_batch(
*read_batch_start = ctxt->offset;
*read_batch_len = ctxt->data_file_size - ctxt->offset;
- if (*read_batch_len > ctxt->buffer_capacity) {
+ if (*read_batch_len > (ib_int64_t)ctxt->buffer_capacity) {
*read_batch_len = ctxt->buffer_capacity;
}
@@ -128,7 +128,7 @@ rf_bitmap_get_next_batch(
{
ulint start_page_id;
- start_page_id = ctxt->offset / ctxt->page_size;
+ start_page_id = (ulint)(ctxt->offset / ctxt->page_size);
xb_a (ctxt->offset % ctxt->page_size == 0);
@@ -170,7 +170,7 @@ rf_bitmap_get_next_batch(
buffer capacity. The subsequent invocations will continue returning
the current block in buffer-sized pieces until ctxt->filter_batch_end
is reached, trigerring the next bitmap query. */
- if (*read_batch_len > ctxt->buffer_capacity) {
+ if (*read_batch_len > (ib_int64_t)ctxt->buffer_capacity) {
*read_batch_len = ctxt->buffer_capacity;
}
diff --git a/extra/mariabackup/read_filt.h b/extra/mariabackup/read_filt.h
index 73fef06a288..d16f4e1093d 100644
--- a/extra/mariabackup/read_filt.h
+++ b/extra/mariabackup/read_filt.h
@@ -33,14 +33,14 @@ struct xb_fil_cur_t;
struct xb_read_filt_ctxt_t {
ib_int64_t offset; /*!< current file offset */
ib_int64_t data_file_size; /*!< data file size */
- ib_int64_t buffer_capacity;/*!< read buffer capacity */
- ulint space_id; /*!< space id */
+ size_t buffer_capacity;/*!< read buffer capacity */
+ ib_int64_t space_id; /*!< space id */
/* The following fields used only in bitmap filter */
/* Move these to union if any other filters are added in future */
xb_page_bitmap_range *bitmap_range; /*!< changed page bitmap range
iterator for space_id */
- ulint page_size; /*!< page size */
- ulint filter_batch_end;/*!< the ending page id of the
+ size_t page_size; /*!< page size */
+ ulint filter_batch_end;/*!< the ending page id of the
current changed page block in
the bitmap */
};
diff --git a/extra/mariabackup/version_check.pl b/extra/mariabackup/version_check.pl
deleted file mode 100644
index 865e2eacb4a..00000000000
--- a/extra/mariabackup/version_check.pl
+++ /dev/null
@@ -1,1373 +0,0 @@
-use warnings FATAL => 'all';
-use strict;
-use English qw(-no_match_vars);
-use POSIX "strftime";
-
-my @required_perl_version = (5, 0, 5);
-my $required_perl_version_old_style = 5.005;
-
-# check existence of DBD::mysql module
-eval {
- require DBD::mysql;
-};
-my $dbd_mysql_installed = $EVAL_ERROR ? 0 : 1;
-
-my $now;
-my %mysql;
-my $prefix = "version_check";
-
-
-# ###########################################################################
-# HTTPMicro package
-# This package is a copy without comments from the original. The original
-# with comments and its test file can be found in the Bazaar repository at,
-# lib/HTTPMicro.pm
-# t/lib/HTTPMicro.t
-# See https://launchpad.net/percona-toolkit for more information.
-# ###########################################################################
-{
-
-package HTTPMicro;
-BEGIN {
- $HTTPMicro::VERSION = '0.001';
-}
-use strict;
-use warnings;
-
-use Carp ();
-
-
-my @attributes;
-BEGIN {
- @attributes = qw(agent timeout);
- no strict 'refs';
- for my $accessor ( @attributes ) {
- *{$accessor} = sub {
- @_ > 1 ? $_[0]->{$accessor} = $_[1] : $_[0]->{$accessor};
- };
- }
-}
-
-sub new {
- my($class, %args) = @_;
- (my $agent = $class) =~ s{::}{-}g;
- my $self = {
- agent => $agent . "/" . ($class->VERSION || 0),
- timeout => 60,
- };
- for my $key ( @attributes ) {
- $self->{$key} = $args{$key} if exists $args{$key}
- }
- return bless $self, $class;
-}
-
-my %DefaultPort = (
- http => 80,
- https => 443,
-);
-
-sub request {
- my ($self, $method, $url, $args) = @_;
- @_ == 3 || (@_ == 4 && ref $args eq 'HASH')
- or Carp::croak(q/Usage: $http->request(METHOD, URL, [HASHREF])/);
- $args ||= {}; # we keep some state in this during _request
-
- my $response;
- for ( 0 .. 1 ) {
- $response = eval { $self->_request($method, $url, $args) };
- last unless $@ && $method eq 'GET'
- && $@ =~ m{^(?:Socket closed|Unexpected end)};
- }
-
- if (my $e = "$@") {
- $response = {
- success => q{},
- status => 599,
- reason => 'Internal Exception',
- content => $e,
- headers => {
- 'content-type' => 'text/plain',
- 'content-length' => length $e,
- }
- };
- }
- return $response;
-}
-
-sub _request {
- my ($self, $method, $url, $args) = @_;
-
- my ($scheme, $host, $port, $path_query) = $self->_split_url($url);
-
- my $request = {
- method => $method,
- scheme => $scheme,
- host_port => ($port == $DefaultPort{$scheme} ? $host : "$host:$port"),
- uri => $path_query,
- headers => {},
- };
-
- my $handle = HTTPMicro::Handle->new(timeout => $self->{timeout});
-
- $handle->connect($scheme, $host, $port);
-
- $self->_prepare_headers_and_cb($request, $args);
- $handle->write_request_header(@{$request}{qw/method uri headers/});
- $handle->write_content_body($request) if $request->{content};
-
- my $response;
- do { $response = $handle->read_response_header }
- until (substr($response->{status},0,1) ne '1');
-
- if (!($method eq 'HEAD' || $response->{status} =~ /^[23]04/)) {
- $response->{content} = '';
- $handle->read_content_body(sub { $_[1]->{content} .= $_[0] }, $response);
- }
-
- $handle->close;
- $response->{success} = substr($response->{status},0,1) eq '2';
- return $response;
-}
-
-sub _prepare_headers_and_cb {
- my ($self, $request, $args) = @_;
-
- for ($args->{headers}) {
- next unless defined;
- while (my ($k, $v) = each %$_) {
- $request->{headers}{lc $k} = $v;
- }
- }
- $request->{headers}{'host'} = $request->{host_port};
- $request->{headers}{'connection'} = "close";
- $request->{headers}{'user-agent'} ||= $self->{agent};
-
- if (defined $args->{content}) {
- $request->{headers}{'content-type'} ||= "application/octet-stream";
- utf8::downgrade($args->{content}, 1)
- or Carp::croak(q/Wide character in request message body/);
- $request->{headers}{'content-length'} = length $args->{content};
- $request->{content} = $args->{content};
- }
- return;
-}
-
-sub _split_url {
- my $url = pop;
-
- my ($scheme, $authority, $path_query) = $url =~ m<\A([^:/?#]+)://([^/?#]*)([^#]*)>
- or Carp::croak(qq/Cannot parse URL: '$url'/);
-
- $scheme = lc $scheme;
- $path_query = "/$path_query" unless $path_query =~ m<\A/>;
-
- my $host = (length($authority)) ? lc $authority : 'localhost';
- $host =~ s/\A[^@]*@//; # userinfo
- my $port = do {
- $host =~ s/:([0-9]*)\z// && length $1
- ? $1
- : $DefaultPort{$scheme}
- };
-
- return ($scheme, $host, $port, $path_query);
-}
-
-package
- HTTPMicro::Handle; # hide from PAUSE/indexers
-use strict;
-use warnings;
-
-use Carp qw[croak];
-use Errno qw[EINTR EPIPE];
-use IO::Socket qw[SOCK_STREAM];
-
-sub BUFSIZE () { 32768 }
-
-my $Printable = sub {
- local $_ = shift;
- s/\r/\\r/g;
- s/\n/\\n/g;
- s/\t/\\t/g;
- s/([^\x20-\x7E])/sprintf('\\x%.2X', ord($1))/ge;
- $_;
-};
-
-sub new {
- my ($class, %args) = @_;
- return bless {
- rbuf => '',
- timeout => 60,
- max_line_size => 16384,
- %args
- }, $class;
-}
-
-my $ssl_verify_args = {
- check_cn => "when_only",
- wildcards_in_alt => "anywhere",
- wildcards_in_cn => "anywhere"
-};
-
-sub connect {
- @_ == 4 || croak(q/Usage: $handle->connect(scheme, host, port)/);
- my ($self, $scheme, $host, $port) = @_;
-
- if ( $scheme eq 'https' ) {
- eval "require IO::Socket::SSL"
- unless exists $INC{'IO/Socket/SSL.pm'};
- croak(qq/IO::Socket::SSL must be installed for https support\n/)
- unless $INC{'IO/Socket/SSL.pm'};
- }
- elsif ( $scheme ne 'http' ) {
- croak(qq/Unsupported URL scheme '$scheme'\n/);
- }
-
- $self->{fh} = 'IO::Socket::INET'->new(
- PeerHost => $host,
- PeerPort => $port,
- Proto => 'tcp',
- Type => SOCK_STREAM,
- Timeout => $self->{timeout}
- ) or croak(qq/Could not connect to '$host:$port': $@/);
-
- binmode($self->{fh})
- or croak(qq/Could not binmode() socket: '$!'/);
-
- if ( $scheme eq 'https') {
- IO::Socket::SSL->start_SSL($self->{fh});
- ref($self->{fh}) eq 'IO::Socket::SSL'
- or die(qq/SSL connection failed for $host\n/);
- if ( $self->{fh}->can("verify_hostname") ) {
- $self->{fh}->verify_hostname( $host, $ssl_verify_args )
- or die(qq/SSL certificate not valid for $host\n/);
- }
- else {
- my $fh = $self->{fh};
- _verify_hostname_of_cert($host, _peer_certificate($fh), $ssl_verify_args)
- or die(qq/SSL certificate not valid for $host\n/);
- }
- }
-
- $self->{host} = $host;
- $self->{port} = $port;
-
- return $self;
-}
-
-sub close {
- @_ == 1 || croak(q/Usage: $handle->close()/);
- my ($self) = @_;
- CORE::close($self->{fh})
- or croak(qq/Could not close socket: '$!'/);
-}
-
-sub write {
- @_ == 2 || croak(q/Usage: $handle->write(buf)/);
- my ($self, $buf) = @_;
-
- my $len = length $buf;
- my $off = 0;
-
- local $SIG{PIPE} = 'IGNORE';
-
- while () {
- $self->can_write
- or croak(q/Timed out while waiting for socket to become ready for writing/);
- my $r = syswrite($self->{fh}, $buf, $len, $off);
- if (defined $r) {
- $len -= $r;
- $off += $r;
- last unless $len > 0;
- }
- elsif ($! == EPIPE) {
- croak(qq/Socket closed by remote server: $!/);
- }
- elsif ($! != EINTR) {
- croak(qq/Could not write to socket: '$!'/);
- }
- }
- return $off;
-}
-
-sub read {
- @_ == 2 || @_ == 3 || croak(q/Usage: $handle->read(len)/);
- my ($self, $len) = @_;
-
- my $buf = '';
- my $got = length $self->{rbuf};
-
- if ($got) {
- my $take = ($got < $len) ? $got : $len;
- $buf = substr($self->{rbuf}, 0, $take, '');
- $len -= $take;
- }
-
- while ($len > 0) {
- $self->can_read
- or croak(q/Timed out while waiting for socket to become ready for reading/);
- my $r = sysread($self->{fh}, $buf, $len, length $buf);
- if (defined $r) {
- last unless $r;
- $len -= $r;
- }
- elsif ($! != EINTR) {
- croak(qq/Could not read from socket: '$!'/);
- }
- }
- if ($len) {
- croak(q/Unexpected end of stream/);
- }
- return $buf;
-}
-
-sub readline {
- @_ == 1 || croak(q/Usage: $handle->readline()/);
- my ($self) = @_;
-
- while () {
- if ($self->{rbuf} =~ s/\A ([^\x0D\x0A]* \x0D?\x0A)//x) {
- return $1;
- }
- $self->can_read
- or croak(q/Timed out while waiting for socket to become ready for reading/);
- my $r = sysread($self->{fh}, $self->{rbuf}, BUFSIZE, length $self->{rbuf});
- if (defined $r) {
- last unless $r;
- }
- elsif ($! != EINTR) {
- croak(qq/Could not read from socket: '$!'/);
- }
- }
- croak(q/Unexpected end of stream while looking for line/);
-}
-
-sub read_header_lines {
- @_ == 1 || @_ == 2 || croak(q/Usage: $handle->read_header_lines([headers])/);
- my ($self, $headers) = @_;
- $headers ||= {};
- my $lines = 0;
- my $val;
-
- while () {
- my $line = $self->readline;
-
- if ($line =~ /\A ([^\x00-\x1F\x7F:]+) : [\x09\x20]* ([^\x0D\x0A]*)/x) {
- my ($field_name) = lc $1;
- $val = \($headers->{$field_name} = $2);
- }
- elsif ($line =~ /\A [\x09\x20]+ ([^\x0D\x0A]*)/x) {
- $val
- or croak(q/Unexpected header continuation line/);
- next unless length $1;
- $$val .= ' ' if length $$val;
- $$val .= $1;
- }
- elsif ($line =~ /\A \x0D?\x0A \z/x) {
- last;
- }
- else {
- croak(q/Malformed header line: / . $Printable->($line));
- }
- }
- return $headers;
-}
-
-sub write_header_lines {
- (@_ == 2 && ref $_[1] eq 'HASH') || croak(q/Usage: $handle->write_header_lines(headers)/);
- my($self, $headers) = @_;
-
- my $buf = '';
- while (my ($k, $v) = each %$headers) {
- my $field_name = lc $k;
- $field_name =~ /\A [\x21\x23-\x27\x2A\x2B\x2D\x2E\x30-\x39\x41-\x5A\x5E-\x7A\x7C\x7E]+ \z/x
- or croak(q/Invalid HTTP header field name: / . $Printable->($field_name));
- $field_name =~ s/\b(\w)/\u$1/g;
- $buf .= "$field_name: $v\x0D\x0A";
- }
- $buf .= "\x0D\x0A";
- return $self->write($buf);
-}
-
-sub read_content_body {
- @_ == 3 || @_ == 4 || croak(q/Usage: $handle->read_content_body(callback, response, [read_length])/);
- my ($self, $cb, $response, $len) = @_;
- $len ||= $response->{headers}{'content-length'};
-
- croak("No content-length in the returned response, and this "
- . "UA doesn't implement chunking") unless defined $len;
-
- while ($len > 0) {
- my $read = ($len > BUFSIZE) ? BUFSIZE : $len;
- $cb->($self->read($read), $response);
- $len -= $read;
- }
-
- return;
-}
-
-sub write_content_body {
- @_ == 2 || croak(q/Usage: $handle->write_content_body(request)/);
- my ($self, $request) = @_;
- my ($len, $content_length) = (0, $request->{headers}{'content-length'});
-
- $len += $self->write($request->{content});
-
- $len == $content_length
- or croak(qq/Content-Length missmatch (got: $len expected: $content_length)/);
-
- return $len;
-}
-
-sub read_response_header {
- @_ == 1 || croak(q/Usage: $handle->read_response_header()/);
- my ($self) = @_;
-
- my $line = $self->readline;
-
- $line =~ /\A (HTTP\/(0*\d+\.0*\d+)) [\x09\x20]+ ([0-9]{3}) [\x09\x20]+ ([^\x0D\x0A]*) \x0D?\x0A/x
- or croak(q/Malformed Status-Line: / . $Printable->($line));
-
- my ($protocol, $version, $status, $reason) = ($1, $2, $3, $4);
-
- return {
- status => $status,
- reason => $reason,
- headers => $self->read_header_lines,
- protocol => $protocol,
- };
-}
-
-sub write_request_header {
- @_ == 4 || croak(q/Usage: $handle->write_request_header(method, request_uri, headers)/);
- my ($self, $method, $request_uri, $headers) = @_;
-
- return $self->write("$method $request_uri HTTP/1.1\x0D\x0A")
- + $self->write_header_lines($headers);
-}
-
-sub _do_timeout {
- my ($self, $type, $timeout) = @_;
- $timeout = $self->{timeout}
- unless defined $timeout && $timeout >= 0;
-
- my $fd = fileno $self->{fh};
- defined $fd && $fd >= 0
- or croak(q/select(2): 'Bad file descriptor'/);
-
- my $initial = time;
- my $pending = $timeout;
- my $nfound;
-
- vec(my $fdset = '', $fd, 1) = 1;
-
- while () {
- $nfound = ($type eq 'read')
- ? select($fdset, undef, undef, $pending)
- : select(undef, $fdset, undef, $pending) ;
- if ($nfound == -1) {
- $! == EINTR
- or croak(qq/select(2): '$!'/);
- redo if !$timeout || ($pending = $timeout - (time - $initial)) > 0;
- $nfound = 0;
- }
- last;
- }
- $! = 0;
- return $nfound;
-}
-
-sub can_read {
- @_ == 1 || @_ == 2 || croak(q/Usage: $handle->can_read([timeout])/);
- my $self = shift;
- return $self->_do_timeout('read', @_)
-}
-
-sub can_write {
- @_ == 1 || @_ == 2 || croak(q/Usage: $handle->can_write([timeout])/);
- my $self = shift;
- return $self->_do_timeout('write', @_)
-}
-
-my $prog = <<'EOP';
-BEGIN {
- if ( defined &IO::Socket::SSL::CAN_IPV6 ) {
- *CAN_IPV6 = \*IO::Socket::SSL::CAN_IPV6;
- }
- else {
- constant->import( CAN_IPV6 => '' );
- }
- my %const = (
- NID_CommonName => 13,
- GEN_DNS => 2,
- GEN_IPADD => 7,
- );
- while ( my ($name,$value) = each %const ) {
- no strict 'refs';
- *{$name} = UNIVERSAL::can( 'Net::SSLeay', $name ) || sub { $value };
- }
-}
-{
- my %dispatcher = (
- issuer => sub { Net::SSLeay::X509_NAME_oneline( Net::SSLeay::X509_get_issuer_name( shift )) },
- subject => sub { Net::SSLeay::X509_NAME_oneline( Net::SSLeay::X509_get_subject_name( shift )) },
- );
- if ( $Net::SSLeay::VERSION >= 1.30 ) {
- $dispatcher{commonName} = sub {
- my $cn = Net::SSLeay::X509_NAME_get_text_by_NID(
- Net::SSLeay::X509_get_subject_name( shift ), NID_CommonName);
- $cn =~s{\0$}{}; # work around Bug in Net::SSLeay <1.33
- $cn;
- }
- } else {
- $dispatcher{commonName} = sub {
- croak "you need at least Net::SSLeay version 1.30 for getting commonName"
- }
- }
-
- if ( $Net::SSLeay::VERSION >= 1.33 ) {
- $dispatcher{subjectAltNames} = sub { Net::SSLeay::X509_get_subjectAltNames( shift ) };
- } else {
- $dispatcher{subjectAltNames} = sub {
- return;
- };
- }
-
- $dispatcher{authority} = $dispatcher{issuer};
- $dispatcher{owner} = $dispatcher{subject};
- $dispatcher{cn} = $dispatcher{commonName};
-
- sub _peer_certificate {
- my ($self, $field) = @_;
- my $ssl = $self->_get_ssl_object or return;
-
- my $cert = ${*$self}{_SSL_certificate}
- ||= Net::SSLeay::get_peer_certificate($ssl)
- or return $self->error("Could not retrieve peer certificate");
-
- if ($field) {
- my $sub = $dispatcher{$field} or croak
- "invalid argument for peer_certificate, valid are: ".join( " ",keys %dispatcher ).
- "\nMaybe you need to upgrade your Net::SSLeay";
- return $sub->($cert);
- } else {
- return $cert
- }
- }
-
-
- my %scheme = (
- ldap => {
- wildcards_in_cn => 0,
- wildcards_in_alt => 'leftmost',
- check_cn => 'always',
- },
- http => {
- wildcards_in_cn => 'anywhere',
- wildcards_in_alt => 'anywhere',
- check_cn => 'when_only',
- },
- smtp => {
- wildcards_in_cn => 0,
- wildcards_in_alt => 0,
- check_cn => 'always'
- },
- none => {}, # do not check
- );
-
- $scheme{www} = $scheme{http}; # alias
- $scheme{xmpp} = $scheme{http}; # rfc 3920
- $scheme{pop3} = $scheme{ldap}; # rfc 2595
- $scheme{imap} = $scheme{ldap}; # rfc 2595
- $scheme{acap} = $scheme{ldap}; # rfc 2595
- $scheme{nntp} = $scheme{ldap}; # rfc 4642
- $scheme{ftp} = $scheme{http}; # rfc 4217
-
-
- sub _verify_hostname_of_cert {
- my $identity = shift;
- my $cert = shift;
- my $scheme = shift || 'none';
- if ( ! ref($scheme) ) {
- $scheme = $scheme{$scheme} or croak "scheme $scheme not defined";
- }
-
- return 1 if ! %$scheme; # 'none'
-
- my $commonName = $dispatcher{cn}->($cert);
- my @altNames = $dispatcher{subjectAltNames}->($cert);
-
- if ( my $sub = $scheme->{callback} ) {
- return $sub->($identity,$commonName,@altNames);
- }
-
-
- my $ipn;
- if ( CAN_IPV6 and $identity =~m{:} ) {
- $ipn = IO::Socket::SSL::inet_pton(IO::Socket::SSL::AF_INET6,$identity)
- or croak "'$identity' is not IPv6, but neither IPv4 nor hostname";
- } elsif ( $identity =~m{^\d+\.\d+\.\d+\.\d+$} ) {
- $ipn = IO::Socket::SSL::inet_aton( $identity ) or croak "'$identity' is not IPv4, but neither IPv6 nor hostname";
- } else {
- if ( $identity =~m{[^a-zA-Z0-9_.\-]} ) {
- $identity =~m{\0} and croak("name '$identity' has \\0 byte");
- $identity = IO::Socket::SSL::idn_to_ascii($identity) or
- croak "Warning: Given name '$identity' could not be converted to IDNA!";
- }
- }
-
- my $check_name = sub {
- my ($name,$identity,$wtyp) = @_;
- $wtyp ||= '';
- my $pattern;
- if ( $wtyp eq 'anywhere' and $name =~m{^([a-zA-Z0-9_\-]*)\*(.+)} ) {
- $pattern = qr{^\Q$1\E[a-zA-Z0-9_\-]*\Q$2\E$}i;
- } elsif ( $wtyp eq 'leftmost' and $name =~m{^\*(\..+)$} ) {
- $pattern = qr{^[a-zA-Z0-9_\-]*\Q$1\E$}i;
- } else {
- $pattern = qr{^\Q$name\E$}i;
- }
- return $identity =~ $pattern;
- };
-
- my $alt_dnsNames = 0;
- while (@altNames) {
- my ($type, $name) = splice (@altNames, 0, 2);
- if ( $ipn and $type == GEN_IPADD ) {
- return 1 if $ipn eq $name;
-
- } elsif ( ! $ipn and $type == GEN_DNS ) {
- $name =~s/\s+$//; $name =~s/^\s+//;
- $alt_dnsNames++;
- $check_name->($name,$identity,$scheme->{wildcards_in_alt})
- and return 1;
- }
- }
-
- if ( ! $ipn and (
- $scheme->{check_cn} eq 'always' or
- $scheme->{check_cn} eq 'when_only' and !$alt_dnsNames)) {
- $check_name->($commonName,$identity,$scheme->{wildcards_in_cn})
- and return 1;
- }
-
- return 0; # no match
- }
-}
-EOP
-
-eval { require IO::Socket::SSL };
-if ( $INC{"IO/Socket/SSL.pm"} ) {
- eval $prog;
- die $@ if $@;
-}
-
-1;
-}
-# ###########################################################################
-# End HTTPMicro package
-# ###########################################################################
-
-# ###########################################################################
-# VersionCheck package
-# This package is a copy without comments from the original. The original
-# with comments and its test file can be found in the Bazaar repository at,
-# lib/VersionCheck.pm
-# t/lib/VersionCheck.t
-# See https://launchpad.net/percona-toolkit for more information.
-# ###########################################################################
-{
-package VersionCheck;
-
-
-use strict;
-use warnings FATAL => 'all';
-use English qw(-no_match_vars);
-
-use constant PTDEBUG => $ENV{PTDEBUG} || 0;
-
-use Data::Dumper;
-local $Data::Dumper::Indent = 1;
-local $Data::Dumper::Sortkeys = 1;
-local $Data::Dumper::Quotekeys = 0;
-
-use Digest::MD5 qw(md5_hex);
-use Sys::Hostname qw(hostname);
-use File::Basename qw();
-use File::Spec;
-use FindBin qw();
-
-eval {
- require Percona::Toolkit;
- require HTTPMicro;
-};
-
-{
- my $file = 'percona-version-check';
- my $home = $ENV{HOME} || $ENV{HOMEPATH} || $ENV{USERPROFILE} || '.';
- my @vc_dirs = (
- '/etc/percona',
- '/etc/percona-toolkit',
- '/tmp',
- "$home",
- );
-
- if ($ENV{PTDEBUG_VERSION_CHECK_HOME}) {
- @vc_dirs = ( $ENV{PTDEBUG_VERSION_CHECK_HOME} );
- }
-
- sub version_check_file {
- foreach my $dir ( @vc_dirs ) {
- if ( -d $dir && -w $dir ) {
- PTDEBUG && _d('Version check file', $file, 'in', $dir);
- return $dir . '/' . $file;
- }
- }
- PTDEBUG && _d('Version check file', $file, 'in', $ENV{PWD});
- return $file; # in the CWD
- }
-}
-
-sub version_check_time_limit {
- return 60 * 60 * 24; # one day
-}
-
-
-sub version_check {
- my (%args) = @_;
-
- my $instances = $args{instances} || [];
- my $instances_to_check;
-
- PTDEBUG && _d('FindBin::Bin:', $FindBin::Bin);
- if ( !$args{force} ) {
- if ( $FindBin::Bin
- && (-d "$FindBin::Bin/../.bzr" || -d "$FindBin::Bin/../../.bzr") ) {
- PTDEBUG && _d("$FindBin::Bin/../.bzr disables --version-check");
- return;
- }
- }
-
- eval {
- foreach my $instance ( @$instances ) {
- my ($name, $id) = get_instance_id($instance);
- $instance->{name} = $name;
- $instance->{id} = $id;
- }
-
- push @$instances, { name => 'system', id => 0 };
-
- $instances_to_check = get_instances_to_check(
- instances => $instances,
- vc_file => $args{vc_file}, # testing
- now => $args{now}, # testing
- );
- PTDEBUG && _d(scalar @$instances_to_check, 'instances to check');
- return unless @$instances_to_check;
-
- my $protocol = 'https';
- eval { require IO::Socket::SSL; };
- if ( $EVAL_ERROR ) {
- PTDEBUG && _d($EVAL_ERROR);
- PTDEBUG && _d("SSL not available, won't run version_check");
- return;
- }
- PTDEBUG && _d('Using', $protocol);
-
- my $advice = pingback(
- instances => $instances_to_check,
- protocol => $protocol,
- url => $args{url} # testing
- || $ENV{PERCONA_VERSION_CHECK_URL} # testing
- || "$protocol://v.percona.com",
- );
- if ( $advice ) {
- PTDEBUG && _d('Advice:', Dumper($advice));
- if ( scalar @$advice > 1) {
- print "\n# " . scalar @$advice . " software updates are "
- . "available:\n";
- }
- else {
- print "\n# A software update is available:\n";
- }
- print join("\n", map { "# * $_" } @$advice), "\n\n";
- }
- };
- if ( $EVAL_ERROR ) {
- PTDEBUG && _d('Version check failed:', $EVAL_ERROR);
- }
-
- if ( @$instances_to_check ) {
- eval {
- update_check_times(
- instances => $instances_to_check,
- vc_file => $args{vc_file}, # testing
- now => $args{now}, # testing
- );
- };
- if ( $EVAL_ERROR ) {
- PTDEBUG && _d('Error updating version check file:', $EVAL_ERROR);
- }
- }
-
- if ( $ENV{PTDEBUG_VERSION_CHECK} ) {
- warn "Exiting because the PTDEBUG_VERSION_CHECK "
- . "environment variable is defined.\n";
- exit 255;
- }
-
- return;
-}
-
-sub get_instances_to_check {
- my (%args) = @_;
-
- my $instances = $args{instances};
- my $now = $args{now} || int(time);
- my $vc_file = $args{vc_file} || version_check_file();
-
- if ( !-f $vc_file ) {
- PTDEBUG && _d('Version check file', $vc_file, 'does not exist;',
- 'version checking all instances');
- return $instances;
- }
-
- open my $fh, '<', $vc_file or die "Cannot open $vc_file: $OS_ERROR";
- chomp(my $file_contents = do { local $/ = undef; <$fh> });
- PTDEBUG && _d('Version check file', $vc_file, 'contents:', $file_contents);
- close $fh;
- my %last_check_time_for = $file_contents =~ /^([^,]+),(.+)$/mg;
-
- my $check_time_limit = version_check_time_limit();
- my @instances_to_check;
- foreach my $instance ( @$instances ) {
- my $last_check_time = $last_check_time_for{ $instance->{id} };
- PTDEBUG && _d('Intsance', $instance->{id}, 'last checked',
- $last_check_time, 'now', $now, 'diff', $now - ($last_check_time || 0),
- 'hours until next check',
- sprintf '%.2f',
- ($check_time_limit - ($now - ($last_check_time || 0))) / 3600);
- if ( !defined $last_check_time
- || ($now - $last_check_time) >= $check_time_limit ) {
- PTDEBUG && _d('Time to check', Dumper($instance));
- push @instances_to_check, $instance;
- }
- }
-
- return \@instances_to_check;
-}
-
-sub update_check_times {
- my (%args) = @_;
-
- my $instances = $args{instances};
- my $now = $args{now} || int(time);
- my $vc_file = $args{vc_file} || version_check_file();
- PTDEBUG && _d('Updating last check time:', $now);
-
- my %all_instances = map {
- $_->{id} => { name => $_->{name}, ts => $now }
- } @$instances;
-
- if ( -f $vc_file ) {
- open my $fh, '<', $vc_file or die "Cannot read $vc_file: $OS_ERROR";
- my $contents = do { local $/ = undef; <$fh> };
- close $fh;
-
- foreach my $line ( split("\n", ($contents || '')) ) {
- my ($id, $ts) = split(',', $line);
- if ( !exists $all_instances{$id} ) {
- $all_instances{$id} = { ts => $ts }; # original ts, not updated
- }
- }
- }
-
- open my $fh, '>', $vc_file or die "Cannot write to $vc_file: $OS_ERROR";
- foreach my $id ( sort keys %all_instances ) {
- PTDEBUG && _d('Updated:', $id, Dumper($all_instances{$id}));
- print { $fh } $id . ',' . $all_instances{$id}->{ts} . "\n";
- }
- close $fh;
-
- return;
-}
-
-sub get_instance_id {
- my ($instance) = @_;
-
- my $dbh = $instance->{dbh};
- my $dsn = $instance->{dsn};
-
- my $sql = q{SELECT CONCAT(@@hostname, @@port)};
- PTDEBUG && _d($sql);
- my ($name) = eval { $dbh->selectrow_array($sql) };
- if ( $EVAL_ERROR ) {
- PTDEBUG && _d($EVAL_ERROR);
- $sql = q{SELECT @@hostname};
- PTDEBUG && _d($sql);
- ($name) = eval { $dbh->selectrow_array($sql) };
- if ( $EVAL_ERROR ) {
- PTDEBUG && _d($EVAL_ERROR);
- $name = ($dsn->{h} || 'localhost') . ($dsn->{P} || 3306);
- }
- else {
- $sql = q{SHOW VARIABLES LIKE 'port'};
- PTDEBUG && _d($sql);
- my (undef, $port) = eval { $dbh->selectrow_array($sql) };
- PTDEBUG && _d('port:', $port);
- $name .= $port || '';
- }
- }
- my $id = md5_hex($name);
-
- PTDEBUG && _d('MySQL instance:', $id, $name, Dumper($dsn));
-
- return $name, $id;
-}
-
-
-sub pingback {
- my (%args) = @_;
- my @required_args = qw(url instances);
- foreach my $arg ( @required_args ) {
- die "I need a $arg arugment" unless $args{$arg};
- }
- my $url = $args{url};
- my $instances = $args{instances};
-
- my $ua = $args{ua} || HTTPMicro->new( timeout => 3 );
-
- my $response = $ua->request('GET', $url);
- PTDEBUG && _d('Server response:', Dumper($response));
- die "No response from GET $url"
- if !$response;
- die("GET on $url returned HTTP status $response->{status}; expected 200\n",
- ($response->{content} || '')) if $response->{status} != 200;
- die("GET on $url did not return any programs to check")
- if !$response->{content};
-
- my $items = parse_server_response(
- response => $response->{content}
- );
- die "Failed to parse server requested programs: $response->{content}"
- if !scalar keys %$items;
-
- my $versions = get_versions(
- items => $items,
- instances => $instances,
- );
- die "Failed to get any program versions; should have at least gotten Perl"
- if !scalar keys %$versions;
-
- my $client_content = encode_client_response(
- items => $items,
- versions => $versions,
- general_id => md5_hex( hostname() ),
- );
-
- my $client_response = {
- headers => { "X-Percona-Toolkit-Tool" => File::Basename::basename($0) },
- content => $client_content,
- };
- PTDEBUG && _d('Client response:', Dumper($client_response));
-
- $response = $ua->request('POST', $url, $client_response);
- PTDEBUG && _d('Server suggestions:', Dumper($response));
- die "No response from POST $url $client_response"
- if !$response;
- die "POST $url returned HTTP status $response->{status}; expected 200"
- if $response->{status} != 200;
-
- return unless $response->{content};
-
- $items = parse_server_response(
- response => $response->{content},
- split_vars => 0,
- );
- die "Failed to parse server suggestions: $response->{content}"
- if !scalar keys %$items;
- my @suggestions = map { $_->{vars} }
- sort { $a->{item} cmp $b->{item} }
- values %$items;
-
- return \@suggestions;
-}
-
-sub encode_client_response {
- my (%args) = @_;
- my @required_args = qw(items versions general_id);
- foreach my $arg ( @required_args ) {
- die "I need a $arg arugment" unless $args{$arg};
- }
- my ($items, $versions, $general_id) = @args{@required_args};
-
- my @lines;
- foreach my $item ( sort keys %$items ) {
- next unless exists $versions->{$item};
- if ( ref($versions->{$item}) eq 'HASH' ) {
- my $mysql_versions = $versions->{$item};
- for my $id ( sort keys %$mysql_versions ) {
- push @lines, join(';', $id, $item, $mysql_versions->{$id});
- }
- }
- else {
- push @lines, join(';', $general_id, $item, $versions->{$item});
- }
- }
-
- my $client_response = join("\n", @lines) . "\n";
- return $client_response;
-}
-
-sub parse_server_response {
- my (%args) = @_;
- my @required_args = qw(response);
- foreach my $arg ( @required_args ) {
- die "I need a $arg arugment" unless $args{$arg};
- }
- my ($response) = @args{@required_args};
-
- my %items = map {
- my ($item, $type, $vars) = split(";", $_);
- if ( !defined $args{split_vars} || $args{split_vars} ) {
- $vars = [ split(",", ($vars || '')) ];
- }
- $item => {
- item => $item,
- type => $type,
- vars => $vars,
- };
- } split("\n", $response);
-
- PTDEBUG && _d('Items:', Dumper(\%items));
-
- return \%items;
-}
-
-my %sub_for_type = (
- os_version => \&get_os_version,
- perl_version => \&get_perl_version,
- perl_module_version => \&get_perl_module_version,
- mysql_variable => \&get_mysql_variable,
-);
-
-sub valid_item {
- my ($item) = @_;
- return unless $item;
- if ( !exists $sub_for_type{ $item->{type} } ) {
- PTDEBUG && _d('Invalid type:', $item->{type});
- return 0;
- }
- return 1;
-}
-
-sub get_versions {
- my (%args) = @_;
- my @required_args = qw(items);
- foreach my $arg ( @required_args ) {
- die "I need a $arg arugment" unless $args{$arg};
- }
- my ($items) = @args{@required_args};
-
- my %versions;
- foreach my $item ( values %$items ) {
- next unless valid_item($item);
- eval {
- my $version = $sub_for_type{ $item->{type} }->(
- item => $item,
- instances => $args{instances},
- );
- if ( $version ) {
- chomp $version unless ref($version);
- $versions{$item->{item}} = $version;
- }
- };
- if ( $EVAL_ERROR ) {
- PTDEBUG && _d('Error getting version for', Dumper($item), $EVAL_ERROR);
- }
- }
-
- return \%versions;
-}
-
-
-sub get_os_version {
- if ( $OSNAME eq 'MSWin32' ) {
- require Win32;
- return Win32::GetOSDisplayName();
- }
-
- chomp(my $platform = `uname -s`);
- PTDEBUG && _d('platform:', $platform);
- return $OSNAME unless $platform;
-
- chomp(my $lsb_release
- = `which lsb_release 2>/dev/null | awk '{print \$1}'` || '');
- PTDEBUG && _d('lsb_release:', $lsb_release);
-
- my $release = "";
-
- if ( $platform eq 'Linux' ) {
- if ( -f "/etc/fedora-release" ) {
- $release = `cat /etc/fedora-release`;
- }
- elsif ( -f "/etc/redhat-release" ) {
- $release = `cat /etc/redhat-release`;
- }
- elsif ( -f "/etc/system-release" ) {
- $release = `cat /etc/system-release`;
- }
- elsif ( $lsb_release ) {
- $release = `$lsb_release -ds`;
- }
- elsif ( -f "/etc/lsb-release" ) {
- $release = `grep DISTRIB_DESCRIPTION /etc/lsb-release`;
- $release =~ s/^\w+="([^"]+)".+/$1/;
- }
- elsif ( -f "/etc/debian_version" ) {
- chomp(my $rel = `cat /etc/debian_version`);
- $release = "Debian $rel";
- if ( -f "/etc/apt/sources.list" ) {
- chomp(my $code_name = `awk '/^deb/ {print \$3}' /etc/apt/sources.list | awk -F/ '{print \$1}'| awk 'BEGIN {FS="|"} {print \$1}' | sort | uniq -c | sort -rn | head -n1 | awk '{print \$2}'`);
- $release .= " ($code_name)" if $code_name;
- }
- }
- elsif ( -f "/etc/os-release" ) { # openSUSE
- chomp($release = `grep PRETTY_NAME /etc/os-release`);
- $release =~ s/^PRETTY_NAME="(.+)"$/$1/;
- }
- elsif ( `ls /etc/*release 2>/dev/null` ) {
- if ( `grep DISTRIB_DESCRIPTION /etc/*release 2>/dev/null` ) {
- $release = `grep DISTRIB_DESCRIPTION /etc/*release | head -n1`;
- }
- else {
- $release = `cat /etc/*release | head -n1`;
- }
- }
- }
- elsif ( $platform =~ m/(?:BSD|^Darwin)$/ ) {
- my $rel = `uname -r`;
- $release = "$platform $rel";
- }
- elsif ( $platform eq "SunOS" ) {
- my $rel = `head -n1 /etc/release` || `uname -r`;
- $release = "$platform $rel";
- }
-
- if ( !$release ) {
- PTDEBUG && _d('Failed to get the release, using platform');
- $release = $platform;
- }
- chomp($release);
-
- $release =~ s/^"|"$//g;
-
- PTDEBUG && _d('OS version =', $release);
- return $release;
-}
-
-sub get_perl_version {
- my (%args) = @_;
- my $item = $args{item};
- return unless $item;
-
- my $version = sprintf '%vd', $PERL_VERSION;
- PTDEBUG && _d('Perl version', $version);
- return $version;
-}
-
-sub get_perl_module_version {
- my (%args) = @_;
- my $item = $args{item};
- return unless $item;
-
- my $var = '$' . $item->{item} . '::VERSION';
- my $version = eval "use $item->{item}; $var;";
- PTDEBUG && _d('Perl version for', $var, '=', $version);
- return $version;
-}
-
-sub get_mysql_variable {
- return get_from_mysql(
- show => 'VARIABLES',
- @_,
- );
-}
-
-sub get_from_mysql {
- my (%args) = @_;
- my $show = $args{show};
- my $item = $args{item};
- my $instances = $args{instances};
- return unless $show && $item;
-
- if ( !$instances || !@$instances ) {
- PTDEBUG && _d('Cannot check', $item,
- 'because there are no MySQL instances');
- return;
- }
-
- if ($item->{item} eq 'MySQL' && $item->{type} eq 'mysql_variable') {
- $item->{vars} = ['version_comment', 'version'];
- }
-
- my @versions;
- my %version_for;
- foreach my $instance ( @$instances ) {
- next unless $instance->{id}; # special system instance has id=0
- my $dbh = $instance->{dbh};
- local $dbh->{FetchHashKeyName} = 'NAME_lc';
- my $sql = qq/SHOW $show/;
- PTDEBUG && _d($sql);
- my $rows = $dbh->selectall_hashref($sql, 'variable_name');
-
- my @versions;
- foreach my $var ( @{$item->{vars}} ) {
- $var = lc($var);
- my $version = $rows->{$var}->{value};
- PTDEBUG && _d('MySQL version for', $item->{item}, '=', $version,
- 'on', $instance->{name});
- push @versions, $version;
- }
- $version_for{ $instance->{id} } = join(' ', @versions);
- }
-
- return \%version_for;
-}
-
-sub _d {
- my ($package, undef, $line) = caller 0;
- @_ = map { (my $temp = $_) =~ s/\n/\n# /g; $temp; }
- map { defined $_ ? $_ : 'undef' }
- @_;
- print STDERR "# $package:$line $PID ", join(' ', @_), "\n";
-}
-
-1;
-}
-# ###########################################################################
-# End VersionCheck package
-# ###########################################################################
-
-#
-# parse_connection_options() subroutine parses connection-related command line
-# options
-#
-sub parse_connection_options {
- my $con = shift;
-
- $con->{dsn} = 'dbi:mysql:';
-
- # this option has to be first
- if ($ENV{option_defaults_file}) {
- $con->{dsn} .= ";mysql_read_default_file=$ENV{option_defaults_file}";
- }
-
- if ($ENV{option_defaults_extra_file}) {
- $con->{dsn} .= ";mysql_read_default_file=$ENV{option_defaults_extra_file}";
- }
-
- $con->{dsn} .= ";mysql_read_default_group=xtrabackup";
-
- if ($ENV{option_mysql_password}) {
- $con->{dsn_password} = "$ENV{option_mysql_password}";
- }
- if ($ENV{option_mysql_user}) {
- $con->{dsn_user} = "$ENV{option_mysql_user}";
- }
- if ($ENV{option_mysql_host}) {
- $con->{dsn} .= ";host=$ENV{option_mysql_host}";
- }
- if ($ENV{option_mysql_port}) {
- $con->{dsn} .= ";port=$ENV{option_mysql_port}";
- }
- if ($ENV{option_mysql_socket}) {
- $con->{dsn} .= ";mysql_socket=$ENV{option_mysql_socket}";
- }
-}
-
-#
-# mysql_connect subroutine connects to MySQL server
-#
-sub mysql_connect {
- my %con;
- my %args = (
- # Defaults
- abort_on_error => 1,
- @_
- );
-
- $con{abort_on_error} = $args{abort_on_error};
-
- parse_connection_options(\%con);
-
- $now = current_time();
- print STDERR "$now $prefix Connecting to MySQL server with DSN '$con{dsn}'" .
- (defined($con{dsn_user}) ? " as '$con{dsn_user}' " : "") .
- " (using password: ";
- if (defined($con{dsn_password})) {
- print STDERR "YES).\n";
- } else {
- print STDERR "NO).\n";
- }
-
- eval {
- $con{dbh}=DBI->connect($con{dsn}, $con{dsn_user},
- $con{dsn_password}, { RaiseError => 1 });
- };
-
- if ($EVAL_ERROR) {
- $con{connect_error}=$EVAL_ERROR;
- } else {
- $now = current_time();
- print STDERR "$now $prefix Connected to MySQL server\n";
- }
-
- if ($args{abort_on_error}) {
- if (!$dbd_mysql_installed) {
- die "Failed to connect to MySQL server as " .
- "DBD::mysql module is not installed";
- } else {
- if (!$con{dbh}) {
- die "Failed to connect to MySQL server: " .
- $con{connect_error};
- }
- }
- }
-
- if ($con{dbh}) {
- $con{dbh}->do("SET SESSION wait_timeout=2147483");
- }
-
- return %con;
-}
-
-#
-# return current local time as string in form "070816 12:23:15"
-#
-sub current_time {
- return strftime("%y%m%d %H:%M:%S", localtime());
-}
-
-
-%mysql = mysql_connect(abort_on_error => 1);
-
-$now = current_time();
-print STDERR
- "$now $prefix Executing a version check against the server...\n";
-
-# Redirect STDOUT to STDERR, as VersionCheck prints alerts to STDOUT
-select STDERR;
-
-VersionCheck::version_check(
- force => 1,
- instances => [ {
- dbh => $mysql{dbh},
- dsn => $mysql{dsn}
- }
- ]
- );
-# Restore STDOUT as the default filehandle
-select STDOUT;
-
-$now = current_time();
-print STDERR "$now $prefix Done.\n";
diff --git a/extra/mariabackup/write_filt.cc b/extra/mariabackup/write_filt.cc
index 129302d7fa0..cf7753bf380 100644
--- a/extra/mariabackup/write_filt.cc
+++ b/extra/mariabackup/write_filt.cc
@@ -75,7 +75,7 @@ wf_incremental_init(xb_write_filt_ctxt_t *ctxt, char *dst_name,
ctxt->cursor = cursor;
/* allocate buffer for incremental backup (4096 pages) */
- buf_size = (UNIV_PAGE_SIZE_MAX / 4 + 1) * UNIV_PAGE_SIZE_MAX;
+ buf_size = (cursor->page_size / 4 + 1) * cursor->page_size;
cp->delta_buf_base = static_cast<byte *>(ut_malloc(buf_size));
memset(cp->delta_buf_base, 0, buf_size);
cp->delta_buf = static_cast<byte *>
diff --git a/extra/mariabackup/write_filt.h b/extra/mariabackup/write_filt.h
index 20213b6f523..bcab263f1dd 100644
--- a/extra/mariabackup/write_filt.h
+++ b/extra/mariabackup/write_filt.h
@@ -27,7 +27,6 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
#include "fil_cur.h"
#include "datasink.h"
-#include "compact.h"
/* Incremental page filter context */
typedef struct {
@@ -41,7 +40,6 @@ typedef struct {
xb_fil_cur_t *cursor;
union {
xb_wf_incremental_ctxt_t wf_incremental_ctxt;
- xb_wf_compact_ctxt_t wf_compact_ctxt;
} u;
} xb_write_filt_ctxt_t;
@@ -56,6 +54,5 @@ typedef struct {
extern xb_write_filt_t wf_write_through;
extern xb_write_filt_t wf_incremental;
-extern xb_write_filt_t wf_compact;
#endif /* XB_WRITE_FILT_H */
diff --git a/extra/mariabackup/wsrep.cc b/extra/mariabackup/wsrep.cc
index 420ada75f36..be11e058255 100644
--- a/extra/mariabackup/wsrep.cc
+++ b/extra/mariabackup/wsrep.cc
@@ -46,7 +46,7 @@ permission notice:
#include <trx0sys.h>
#include "common.h"
-
+#ifdef WITH_WSREP
#define WSREP_XID_PREFIX "WSREPXid"
#define WSREP_XID_PREFIX_LEN MYSQL_XID_PREFIX_LEN
#define WSREP_XID_UUID_OFFSET 8
@@ -61,11 +61,11 @@ permission notice:
/* Galera UUID type - for all unique IDs */
typedef struct wsrep_uuid {
- uint8_t data[16];
+ unsigned char data[16];
} wsrep_uuid_t;
/* sequence number of a writeset, etc. */
-typedef int64_t wsrep_seqno_t;
+typedef long long wsrep_seqno_t;
/* Undefined UUID */
static const wsrep_uuid_t WSREP_UUID_UNDEFINED = {{0,}};
@@ -217,3 +217,4 @@ xb_write_galera_info(bool incremental_prepare)
fclose(fp);
}
+#endif
diff --git a/extra/mariabackup/xb0xb.h b/extra/mariabackup/xb0xb.h
new file mode 100644
index 00000000000..659ab8ea5d0
--- /dev/null
+++ b/extra/mariabackup/xb0xb.h
@@ -0,0 +1,78 @@
+/******************************************************
+Copyright (c) 2012 Percona LLC and/or its affiliates.
+
+Declarations of XtraBackup functions called by InnoDB code.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; version 2 of the License.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
+
+*******************************************************/
+
+#ifndef xb0xb_h
+#define xb0xb_h
+
+
+extern void os_io_init_simple(void);
+extern os_file_t files[1000];
+extern const char *innodb_checksum_algorithm_names[];
+extern TYPELIB innodb_checksum_algorithm_typelib;
+extern dberr_t open_or_create_data_files(
+ ibool* create_new_db,
+#ifdef UNIV_LOG_ARCHIVE
+ lsn_t* min_arch_log_no,
+ lsn_t* max_arch_log_no,
+#endif
+ lsn_t* min_flushed_lsn,
+ lsn_t* max_flushed_lsn,
+ ulint* sum_of_new_sizes)
+ ;
+int
+fil_file_readdir_next_file(
+/*=======================*/
+dberr_t* err, /*!< out: this is set to DB_ERROR if an error
+ was encountered, otherwise not changed */
+ const char* dirname,/*!< in: directory name or path */
+ os_file_dir_t dir, /*!< in: directory stream */
+ os_file_stat_t* info) /*!< in/out: buffer where the
+ info is returned */;
+buf_block_t* btr_node_ptr_get_child(
+ const rec_t* node_ptr,/*!< in: node pointer */
+ dict_index_t* index, /*!< in: index */
+ const ulint* offsets,/*!< in: array returned by rec_get_offsets() */
+ mtr_t* mtr) /*!< in: mtr */;
+
+buf_block_t*
+btr_root_block_get(
+/*===============*/
+const dict_index_t* index, /*!< in: index tree */
+ulint mode, /*!< in: either RW_S_LATCH
+ or RW_X_LATCH */
+ mtr_t* mtr) /*!< in: mtr */;
+fil_space_t*
+fil_space_get_by_name(const char *);
+ibool
+recv_check_cp_is_consistent(const byte* buf);
+void
+innodb_log_checksum_func_update(
+/*============================*/
+ulint algorithm) /*!< in: algorithm */;
+dberr_t recv_find_max_checkpoint(log_group_t** max_group, ulint* max_field);
+dberr_t
+srv_undo_tablespaces_init(
+/*======================*/
+ibool create_new_db,
+ibool backup_mode,
+const ulint n_conf_tablespaces,
+ulint* n_opened);
+
+#endif
diff --git a/extra/mariabackup/xb_regex.h b/extra/mariabackup/xb_regex.h
index 94c5e2a5fa7..2e07e434e27 100644
--- a/extra/mariabackup/xb_regex.h
+++ b/extra/mariabackup/xb_regex.h
@@ -22,50 +22,27 @@ my_regex is used on Windows and native calls are used on POSIX platforms. */
#ifndef XB_REGEX_H
#define XB_REGEX_H
-#ifdef _WIN32
-
-#include <my_regex.h>
-
-typedef my_regex_t xb_regex_t;
-typedef my_regmatch_t xb_regmatch_t;
-
-#define xb_regex_init() my_regex_init(&my_charset_latin1)
-
-#define xb_regexec(preg,string,nmatch,pmatch,eflags) \
- my_regexec(preg, string, nmatch, pmatch, eflags)
-
-#define xb_regerror(errcode,preg,errbuf,errbuf_size) \
- my_regerror(errcode, preg, errbuf, errbuf_size)
-
-#define xb_regcomp(preg,regex,cflags) \
- my_regcomp(preg, regex, cflags, &my_charset_latin1)
-
-#define xb_regfree(preg) my_regfree(preg)
-
-#define xb_regex_end() my_regex_end()
-
-#else /* ! _WIN32 */
-
+#ifdef HAVE_SYSTEM_REGEX
#include <regex.h>
+#else
+#include <pcreposix.h>
+#endif
-typedef regex_t xb_regex_t;
-typedef regmatch_t xb_regmatch_t;
+typedef regex_t* xb_regex_t;
-#define xb_regex_init() do { } while(0)
+#define xb_regex_init()
-#define xb_regexec(preg,string,nmatch,pmatch,eflags) \
+#define xb_regexec(preg,string,nmatch,pmatch,eflags) \
regexec(preg, string, nmatch, pmatch, eflags)
-#define xb_regerror(errcode,preg,errbuf,errbuf_size) \
+#define xb_regerror(errcode,preg,errbuf,errbuf_size) \
regerror(errcode, preg, errbuf, errbuf_size)
-#define xb_regcomp(preg,regex,cflags) \
+#define xb_regcomp(preg,regex,cflags) \
regcomp(preg, regex, cflags)
#define xb_regfree(preg) regfree(preg)
-#define xb_regex_end() do { } while (0)
-
-#endif /* _WIN32 */
+#define xb_regex_end()
#endif /* XB_REGEX_H */
diff --git a/extra/mariabackup/xbcrypt_common.c b/extra/mariabackup/xbcrypt_common.c
index fe34fcb3bb0..52fa2ce3589 100644
--- a/extra/mariabackup/xbcrypt_common.c
+++ b/extra/mariabackup/xbcrypt_common.c
@@ -27,6 +27,7 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
# pragma GCC diagnostic ignored "-Wdeprecated-declarations"
#endif
+#ifdef HAVE_GRYPT
#include <gcrypt.h>
#if GCC_VERSION >= 4002
@@ -58,3 +59,4 @@ xb_crypt_create_iv(void* ivbuf, size_t ivlen)
{
gcry_create_nonce(ivbuf, ivlen);
}
+#endif \ No newline at end of file
diff --git a/extra/mariabackup/xbcrypt_write.c b/extra/mariabackup/xbcrypt_write.c
index 5cbeb67f227..6cc30852215 100644
--- a/extra/mariabackup/xbcrypt_write.c
+++ b/extra/mariabackup/xbcrypt_write.c
@@ -73,7 +73,7 @@ int xb_crypt_write_chunk(xb_wcrypt_t *crypt, const void *buf, size_t olen,
int8store(ptr, (ulonglong)elen); /* encrypted (actual) size */
ptr += 8;
- checksum = crc32(0, buf, elen);
+ checksum = crc32(0, buf, (uint)elen);
int4store(ptr, checksum); /* checksum */
ptr += 4;
diff --git a/extra/mariabackup/xbstream.c b/extra/mariabackup/xbstream.c
index ba3412a359b..9990b00ea4b 100644
--- a/extra/mariabackup/xbstream.c
+++ b/extra/mariabackup/xbstream.c
@@ -208,15 +208,15 @@ int
stream_one_file(File file, xb_wstream_file_t *xbfile)
{
uchar *buf;
- size_t bytes;
- size_t offset;
+ ssize_t bytes;
+ my_off_t offset;
posix_fadvise(file, 0, 0, POSIX_FADV_SEQUENTIAL);
offset = my_tell(file, MYF(MY_WME));
buf = (uchar*)(my_malloc(XBSTREAM_BUFFER_SIZE, MYF(MY_FAE)));
- while ((bytes = my_read(file, buf, XBSTREAM_BUFFER_SIZE,
+ while ((bytes = (ssize_t)my_read(file, buf, XBSTREAM_BUFFER_SIZE,
MYF(MY_WME))) > 0) {
if (xb_stream_write_data(xbfile, buf, bytes)) {
msg("%s: xb_stream_write_data() failed.\n",
@@ -232,7 +232,7 @@ stream_one_file(File file, xb_wstream_file_t *xbfile)
my_free(buf);
- if (bytes == (size_t) -1) {
+ if (bytes < 0) {
return 1;
}
diff --git a/extra/mariabackup/xbstream_read.c b/extra/mariabackup/xbstream_read.c
index 0ffcabd9270..f8d88926ec2 100644
--- a/extra/mariabackup/xbstream_read.c
+++ b/extra/mariabackup/xbstream_read.c
@@ -48,13 +48,13 @@ xb_stream_read_new(void)
stream->buffer = my_malloc(INIT_BUFFER_LEN, MYF(MY_FAE));
stream->buflen = INIT_BUFFER_LEN;
- stream->fd = fileno(stdin);
- stream->offset = 0;
-
#ifdef __WIN__
- setmode(stream->fd, _O_BINARY);
+ setmode(fileno(stdin), _O_BINARY);
#endif
+ stream->fd = my_fileno(stdin);
+ stream->offset = 0;
+
return stream;
}
diff --git a/extra/mariabackup/xbstream_write.c b/extra/mariabackup/xbstream_write.c
index 7b042eea49c..a11811dc375 100644
--- a/extra/mariabackup/xbstream_write.c
+++ b/extra/mariabackup/xbstream_write.c
@@ -34,7 +34,7 @@ struct xb_wstream_struct {
struct xb_wstream_file_struct {
xb_wstream_t *stream;
char *path;
- ulong path_len;
+ size_t path_len;
char chunk[XB_STREAM_MIN_CHUNK_SIZE];
char *chunk_ptr;
size_t chunk_free;
@@ -54,7 +54,7 @@ xb_stream_default_write_callback(xb_wstream_file_t *file __attribute__((unused))
void *userdata __attribute__((unused)),
const void *buf, size_t len)
{
- if (my_write(fileno(stdout), buf, len, MYF(MY_WME | MY_NABP)))
+ if (my_write(my_fileno(stdout), buf, len, MYF(MY_WME | MY_NABP)))
return -1;
return len;
}
@@ -77,7 +77,7 @@ xb_stream_write_open(xb_wstream_t *stream, const char *path,
xb_stream_write_callback *onwrite)
{
xb_wstream_file_t *file;
- ulong path_len;
+ size_t path_len;
path_len = strlen(path);
@@ -90,7 +90,19 @@ xb_stream_write_open(xb_wstream_t *stream, const char *path,
path_len + 1, MYF(MY_FAE));
file->path = (char *) (file + 1);
+#ifdef _WIN32
+ /* Normalize path on Windows, so we can restore elsewhere.*/
+ {
+ int i;
+ for (i = 0; ; i++) {
+ file->path[i] = (path[i] == '\\') ? '/' : path[i];
+ if (!path[i])
+ break;
+ }
+ }
+#else
memcpy(file->path, path, path_len + 1);
+#endif
file->path_len = path_len;
file->stream = stream;
@@ -208,7 +220,7 @@ xb_stream_write_chunk(xb_wstream_file_t *file, const void *buf, size_t len)
int8store(ptr, file->offset); /* Payload offset */
ptr += 8;
- checksum = crc32(0, buf, len); /* checksum */
+ checksum = crc32(0, buf, (uint)len); /* checksum */
int4store(ptr, checksum);
ptr += 4;
diff --git a/extra/mariabackup/xtrabackup.cc b/extra/mariabackup/xtrabackup.cc
index d24c915bb48..3ca6efaafcf 100644
--- a/extra/mariabackup/xtrabackup.cc
+++ b/extra/mariabackup/xtrabackup.cc
@@ -53,9 +53,9 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#ifdef __linux__
# include <sys/prctl.h>
+#include <sys/resource.h>
#endif
-#include <sys/resource.h>
#include <btr0sea.h>
#include <dict0priv.h>
@@ -74,7 +74,6 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#define G_PTR uchar*
#include "common.h"
-#include "xtrabackup_version.h"
#include "datasink.h"
#include "xb_regex.h"
@@ -91,6 +90,10 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#include "backup_mysql.h"
#include "backup_copy.h"
#include "backup_mysql.h"
+#include "xb0xb.h"
+#include "encryption_plugin.h"
+#include <sql_plugin.h>
+#include <srv0srv.h>
/* TODO: replace with appropriate macros used in InnoDB 5.6 */
#define PAGE_ZIP_MIN_SIZE_SHIFT 10
@@ -143,15 +146,16 @@ lsn_t xtrabackup_archived_to_lsn = 0; /* for --archived-to-lsn */
char *xtrabackup_tables = NULL;
+char *xtrabackup_tmpdir;
+
/* List of regular expressions for filtering */
typedef struct xb_regex_list_node_struct xb_regex_list_node_t;
struct xb_regex_list_node_struct {
UT_LIST_NODE_T(xb_regex_list_node_t) regex_list;
- xb_regex_t regex;
+ regex_t regex;
};
static UT_LIST_BASE_NODE_T(xb_regex_list_node_t) regex_list;
-
-static xb_regmatch_t tables_regmatch[1];
+static regmatch_t tables_regmatch[1];
char *xtrabackup_tables_file = NULL;
static hash_table_t* tables_hash = NULL;
@@ -210,6 +214,7 @@ ulint xtrabackup_rebuild_threads = 1;
/* sleep interval beetween log copy iterations in log copying thread
in milliseconds (default is 1 second) */
ulint xtrabackup_log_copy_interval = 1000;
+static ulong max_buf_pool_modified_pct;
/* Ignored option (--log) for MySQL option compatibility */
char* log_ignored_opt = NULL;
@@ -294,8 +299,6 @@ it every INNOBASE_WAKE_INTERVAL'th step. */
#define INNOBASE_WAKE_INTERVAL 32
ulong innobase_active_counter = 0;
-ibool srv_compact_backup = FALSE;
-ibool srv_rebuild_indexes = FALSE;
static char *xtrabackup_debug_sync = NULL;
@@ -309,13 +312,15 @@ lsn_t min_flushed_lsn= 0;
lsn_t max_flushed_lsn= 0;
/* The size of archived log file */
-size_t xtrabackup_arch_file_size = 0ULL;
+ib_int64_t xtrabackup_arch_file_size = 0ULL;
/* The minimal LSN of found archived log files */
lsn_t xtrabackup_arch_first_file_lsn = 0ULL;
/* The maximum LSN of found archived log files */
lsn_t xtrabackup_arch_last_file_lsn = 0ULL;
ulong xb_open_files_limit= 0;
+char *xb_plugin_dir;
+char *xb_plugin_load;
my_bool xb_close_files= FALSE;
/* Datasinks */
@@ -574,6 +579,9 @@ enum options_xtrabackup
OPT_XTRA_INCREMENTAL_FORCE_SCAN,
OPT_DEFAULTS_GROUP,
OPT_OPEN_FILES_LIMIT,
+ OPT_PLUGIN_DIR,
+ OPT_PLUGIN_LOAD,
+ OPT_INNODB_ENCRYPT_LOG,
OPT_CLOSE_FILES,
OPT_CORE_FILE,
@@ -682,7 +690,13 @@ struct my_option xb_client_options[] =
0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
{"stream", OPT_XTRA_STREAM, "Stream all backup files to the standard output "
- "in the specified format. Currently the only supported format is 'tar'.",
+ "in the specified format."
+#ifdef HAVE_LIBARCHIVE
+ "Supported formats are 'tar' and 'xbstream'."
+#else
+ "Supported format is 'xbstream'."
+#endif
+ ,
(G_PTR*) &xtrabackup_stream_str, (G_PTR*) &xtrabackup_stream_str, 0, GET_STR,
REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
@@ -1001,12 +1015,6 @@ struct my_option xb_client_options[] =
#include "sslopt-longopts.h"
-#if !defined(HAVE_YASSL)
- {"server-public-key-path", OPT_SERVER_PUBLIC_KEY,
- "File path to the server public RSA key in PEM format.",
- &opt_server_public_key, &opt_server_public_key, 0,
- GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
-#endif
{ 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}
};
@@ -1098,11 +1106,7 @@ Disable with --skip-innodb-doublewrite.", (G_PTR*) &innobase_use_doublewrite,
(G_PTR*) &innobase_file_per_table,
(G_PTR*) &innobase_file_per_table, 0, GET_BOOL, NO_ARG,
FALSE, 0, 0, 0, 0, 0},
- {"innodb_flush_log_at_trx_commit", OPT_INNODB_FLUSH_LOG_AT_TRX_COMMIT,
- "Set to 0 (write and flush once per second), 1 (write and flush at each commit) or 2 (write at commit, flush once per second).",
- (G_PTR*) &srv_flush_log_at_trx_commit,
- (G_PTR*) &srv_flush_log_at_trx_commit,
- 0, GET_ULONG, OPT_ARG, 1, 0, 2, 0, 0, 0},
+
{"innodb_flush_method", OPT_INNODB_FLUSH_METHOD,
"With which method to flush data.", (G_PTR*) &innobase_unix_file_flush_method,
(G_PTR*) &innobase_unix_file_flush_method, 0, GET_STR, REQUIRED_ARG, 0, 0, 0,
@@ -1203,6 +1207,18 @@ Disable with --skip-innodb-doublewrite.", (G_PTR*) &innobase_use_doublewrite,
(G_PTR*) &defaults_group, (G_PTR*) &defaults_group,
0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
+ {"plugin-dir", OPT_PLUGIN_DIR, "Server plugin directory",
+ &xb_plugin_dir, &xb_plugin_dir,
+ 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0 },
+
+ { "plugin-load", OPT_PLUGIN_LOAD, "encrypton plugin to load",
+ &xb_plugin_load, &xb_plugin_load,
+ 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0 },
+
+ { "innodb-encrypt-log", OPT_INNODB_ENCRYPT_LOG, "encrypton plugin to load",
+ &srv_encrypt_log, &srv_encrypt_log,
+ 0, GET_BOOL, REQUIRED_ARG, 0, 0, 0, 0, 0, 0 },
+
{"open_files_limit", OPT_OPEN_FILES_LIMIT, "the maximum number of file "
"descriptors to reserve with setrlimit().",
(G_PTR*) &xb_open_files_limit, (G_PTR*) &xb_open_files_limit, 0, GET_ULONG,
@@ -1276,9 +1292,8 @@ static const char *xb_server_default_groups[]=
static void print_version(void)
{
- msg("%s version %s based on MySQL server %s %s (%s) (revision id: %s)\n",
- my_progname, XTRABACKUP_VERSION, MYSQL_SERVER_VERSION, SYSTEM_TYPE,
- MACHINE_TYPE, XTRABACKUP_REVISION);
+ msg("%s based on MariaDB server %s %s (%s) \n",
+ my_progname, MYSQL_SERVER_VERSION, SYSTEM_TYPE, MACHINE_TYPE);
}
static void usage(void)
@@ -1515,10 +1530,10 @@ xb_init_log_block_size(void)
{
srv_log_block_size = 0;
if (innobase_log_block_size != 512) {
- uint n_shift = get_bit_shift(innobase_log_block_size);;
+ uint n_shift = (uint)get_bit_shift(innobase_log_block_size);;
if (n_shift > 0) {
- srv_log_block_size = (1 << n_shift);
+ srv_log_block_size = (ulint)(1LL << n_shift);
msg("InnoDB: The log block size is set to %lu.\n",
srv_log_block_size);
}
@@ -1541,13 +1556,13 @@ innodb_init_param(void)
static char current_dir[3]; /* Set if using current lib */
my_bool ret;
char *default_path;
-
+ srv_is_being_started = TRUE;
/* === some variables from mysqld === */
memset((G_PTR) &mysql_tmpdir_list, 0, sizeof(mysql_tmpdir_list));
if (init_tmpdir(&mysql_tmpdir_list, opt_mysql_tmpdir))
exit(EXIT_FAILURE);
-
+ xtrabackup_tmpdir = my_tmpdir(&mysql_tmpdir_list);
/* dummy for initialize all_charsets[] */
get_charset_name(0);
@@ -1555,7 +1570,7 @@ innodb_init_param(void)
srv_page_size_shift = 0;
if (innobase_page_size != (1LL << 14)) {
- int n_shift = get_bit_shift((ulint) innobase_page_size);
+ int n_shift = (int)get_bit_shift((ulint) innobase_page_size);
if (n_shift >= 12 && n_shift <= UNIV_PAGE_SIZE_SHIFT_MAX) {
srv_page_size_shift = n_shift;
@@ -1576,8 +1591,6 @@ innodb_init_param(void)
goto error;
}
- srv_fast_checksum = (ibool) innobase_fast_checksum;
-
/* Check that values don't overflow on 32-bit systems. */
if (sizeof(ulint) == 4) {
if (xtrabackup_use_memory > UINT_MAX32) {
@@ -1738,10 +1751,15 @@ mem_free_and_error:
}
btr_search_enabled = (char) innobase_adaptive_hash_index;
+ btr_search_index_num = 1;
os_use_large_pages = (ibool) innobase_use_large_pages;
os_large_page_size = (ulint) innobase_large_page_size;
+ if (!innobase_log_arch_dir) {
+ static char default_dir[3] = "./";
+ srv_arch_dir = default_dir;
+ }
row_rollback_on_timeout = (ibool) innobase_rollback_on_timeout;
srv_file_per_table = (my_bool) innobase_file_per_table;
@@ -1848,7 +1866,7 @@ static my_bool
innodb_init(void)
{
int err;
-
+ srv_is_being_started = TRUE;
err = innobase_start_or_create_for_mysql();
if (err != DB_SUCCESS) {
@@ -1947,12 +1965,6 @@ xtrabackup_read_metadata(char *filename)
}
/* Optional fields */
- if (fscanf(fp, "compact = %d\n", &t) == 1) {
- xtrabackup_compact = (t == 1);
- } else {
- xtrabackup_compact = 0;
- }
-
if (fscanf(fp, "recover_binlog_info = %d\n", &t) == 1) {
recover_binlog_info = (t == 1);
}
@@ -1981,7 +1993,7 @@ xtrabackup_print_metadata(char *buf, size_t buf_len)
metadata_from_lsn,
metadata_to_lsn,
metadata_last_lsn,
- MY_TEST(xtrabackup_compact == TRUE),
+ MY_TEST(false),
MY_TEST(opt_binlog_info == BINLOG_INFO_LOCKLESS));
}
@@ -2166,7 +2178,7 @@ check_if_table_matches_filters(const char *name)
/* Check against regular expressions list */
for (node = UT_LIST_GET_FIRST(regex_list); node;
node = UT_LIST_GET_NEXT(regex_list, node)) {
- regres = xb_regexec(&node->regex, name, 1,
+ regres = regexec(&node->regex, name, 1,
tables_regmatch, 0);
if (regres != REG_NOMATCH) {
@@ -2213,7 +2225,7 @@ check_if_skip_table(
dbname = NULL;
tbname = name;
- while ((ptr = strchr(tbname, SRV_PATH_SEPARATOR)) != NULL) {
+ while ((ptr = strchr(tbname, '/')) != NULL) {
dbname = tbname;
tbname = ptr + 1;
}
@@ -2287,10 +2299,10 @@ xb_get_zip_size(os_file_t file)
ibool success;
ulint space;
- buf = static_cast<byte *>(ut_malloc(2 * UNIV_PAGE_SIZE_MAX));
- page = static_cast<byte *>(ut_align(buf, UNIV_PAGE_SIZE_MAX));
+ buf = static_cast<byte *>(ut_malloc(2 * UNIV_PAGE_SIZE));
+ page = static_cast<byte *>(ut_align(buf, UNIV_PAGE_SIZE));
- success = os_file_read(file, page, 0, UNIV_PAGE_SIZE_MAX);
+ success = os_file_read(file, page, 0, UNIV_PAGE_SIZE);
if (!success) {
goto end;
}
@@ -2392,8 +2404,6 @@ xtrabackup_copy_datafile(fil_node_t* node, uint thread_n)
/* Setup the page write filter */
if (xtrabackup_incremental) {
write_filter = &wf_incremental;
- } else if (xtrabackup_compact) {
- write_filter = &wf_compact;
} else {
write_filter = &wf_write_through;
}
@@ -2484,6 +2494,7 @@ static
void
xtrabackup_choose_lsn_offset(lsn_t start_lsn)
{
+#if SUPPORT_PERCONA_5_5
ulint no, alt_no, expected_no;
ulint blocks_in_group;
lsn_t tmp_offset, end_lsn;
@@ -2566,8 +2577,11 @@ xtrabackup_choose_lsn_offset(lsn_t start_lsn)
decision to choose one over the other. Die just
like a Buridan's ass */
ut_a(lsn_chosen == 1);
+#endif
}
+extern ibool log_block_checksum_is_ok_or_old_format(const byte* block);
+
/*******************************************************//**
Scans log from a buffer and writes new log data to the outpud datasinc.
@return true if success */
@@ -2704,13 +2718,22 @@ xtrabackup_scan_log_recs(
if (!*finished) {
write_size = RECV_SCAN_SIZE;
} else {
- write_size = ut_uint64_align_up(scanned_lsn,
- OS_FILE_LOG_BLOCK_SIZE) - start_lsn;
+ write_size = (ulint)(ut_uint64_align_up(scanned_lsn,
+ OS_FILE_LOG_BLOCK_SIZE) - start_lsn);
if (!is_last && scanned_lsn % OS_FILE_LOG_BLOCK_SIZE) {
write_size -= OS_FILE_LOG_BLOCK_SIZE;
}
}
+ if (write_size == 0) {
+ return(true);
+ }
+
+ if (srv_encrypt_log) {
+ log_encrypt_before_write(scanned_checkpoint_no,
+ log_sys->buf, write_size);
+ }
+
if (ds_write(dst_log_file, log_sys->buf, write_size)) {
msg("xtrabackup: Error: "
"write to logfile failed\n");
@@ -2756,7 +2779,7 @@ xtrabackup_copy_logfile(lsn_t from_lsn, my_bool is_last)
mutex_enter(&log_sys->mutex);
log_group_read_log_seg(LOG_RECOVER, log_sys->buf,
- group, start_lsn, end_lsn);
+ group, start_lsn, end_lsn, false);
if (!xtrabackup_scan_log_recs(group, is_last,
start_lsn, &contiguous_lsn, &group_scanned_lsn,
@@ -3133,6 +3156,7 @@ xb_load_tablespaces(void)
ibool create_new_db;
ulint err;
ulint sum_of_new_sizes;
+ lsn_t min_arch_logno, max_arch_logno;
for (i = 0; i < srv_n_file_io_threads; i++) {
thread_nr[i] = i;
@@ -3144,6 +3168,7 @@ xb_load_tablespaces(void)
os_thread_sleep(200000); /*0.2 sec*/
err = open_or_create_data_files(&create_new_db,
+ &min_arch_logno, &max_arch_logno,
&min_flushed_lsn, &max_flushed_lsn,
&sum_of_new_sizes);
if (err != DB_SUCCESS) {
@@ -3225,19 +3250,9 @@ xb_data_files_close(void)
for (i = 0; i < 1000; i++) {
os_aio_wake_all_threads_at_shutdown();
- os_mutex_enter(os_sync_mutex);
-
if (os_thread_count == 0) {
-
- os_mutex_exit(os_sync_mutex);
-
- os_thread_sleep(10000);
-
break;
}
-
- os_mutex_exit(os_sync_mutex);
-
os_thread_sleep(10000);
}
@@ -3411,7 +3426,7 @@ xb_register_regex(
node = static_cast<xb_regex_list_node_t *>
(ut_malloc(sizeof(xb_regex_list_node_t)));
- ret = xb_regcomp(&node->regex, regex, REG_EXTENDED);
+ ret = regcomp(&node->regex, regex, REG_EXTENDED);
if (ret != 0) {
xb_regerror(ret, &node->regex, errbuf, sizeof(errbuf));
msg("xtrabackup: error: tables regcomp(%s): %s\n",
@@ -3545,7 +3560,7 @@ xb_filters_free()
while (UT_LIST_GET_LEN(regex_list) > 0) {
xb_regex_list_node_t* node = UT_LIST_GET_FIRST(regex_list);
UT_LIST_REMOVE(regex_list, regex_list, node);
- xb_regfree(&node->regex);
+ regfree(&node->regex);
ut_free(node);
}
@@ -3602,7 +3617,7 @@ open_or_create_log_file(
files[i] = os_file_create(innodb_file_log_key, name,
OS_FILE_OPEN, OS_FILE_NORMAL,
- OS_LOG_FILE, &ret);
+ OS_LOG_FILE, &ret,0);
if (ret == FALSE) {
fprintf(stderr, "InnoDB: Error in opening %s\n", name);
@@ -3631,12 +3646,12 @@ open_or_create_log_file(
which is for this log group */
fil_space_create(name,
- 2 * k + SRV_LOG_SPACE_FIRST_ID, 0, FIL_LOG);
+ 2 * k + SRV_LOG_SPACE_FIRST_ID, 0, FIL_LOG, 0, 0);
}
ut_a(fil_validate());
- ut_a(fil_node_create(name, srv_log_file_size,
+ ut_a(fil_node_create(name, (ulint)srv_log_file_size,
2 * k + SRV_LOG_SPACE_FIRST_ID, FALSE));
if (i == 0) {
log_group_init(k, srv_n_log_files,
@@ -3765,7 +3780,7 @@ xtrabackup_backup_func(void)
srv_read_only_mode = TRUE;
srv_backup_mode = TRUE;
- srv_close_files = xb_close_files;
+ srv_close_files = (bool)xb_close_files;
if (srv_close_files)
msg("xtrabackup: warning: close-files specified. Use it "
@@ -3779,7 +3794,7 @@ xtrabackup_backup_func(void)
xb_normalize_init_values();
-#ifndef __WIN__
+
if (srv_file_flush_method_str == NULL) {
/* These are the default options */
srv_unix_file_flush_method = SRV_UNIX_FSYNC;
@@ -3808,13 +3823,14 @@ xtrabackup_backup_func(void)
"innodb_flush_method\n", srv_file_flush_method_str);
exit(EXIT_FAILURE);
}
-#else /* __WIN__ */
+
/* We can only use synchronous unbuffered IO on Windows for now */
if (srv_file_flush_method_str != NULL) {
msg("xtrabackupp: Warning: "
- "ignoring innodb_flush_method = %s on Windows.\n");
+ "ignoring innodb_flush_method = %s on Windows.\n", srv_file_flush_method_str);
}
+#ifdef _WIN32
srv_win_file_flush_method = SRV_WIN_IO_UNBUFFERED;
srv_use_native_aio = FALSE;
#endif
@@ -3836,10 +3852,14 @@ xtrabackup_backup_func(void)
computers */
}
- os_sync_mutex = NULL;
srv_general_init();
ut_crc32_init();
+#ifdef WITH_INNODB_DISALLOW_WRITES
+ srv_allow_writes_event = os_event_create();
+ os_event_set(srv_allow_writes_event);
+#endif
+
xb_filters_init();
{
@@ -3901,7 +3921,7 @@ xtrabackup_backup_func(void)
}
/* create target dir if not exist */
- if (!my_stat(xtrabackup_target_dir,&stat_info,MYF(0))
+ if (!xtrabackup_stream_str && !my_stat(xtrabackup_target_dir,&stat_info,MYF(0))
&& (my_mkdir(xtrabackup_target_dir,0777,MYF(0)) < 0)){
msg("xtrabackup: Error: cannot mkdir %d: %s\n",
my_errno, xtrabackup_target_dir);
@@ -3950,10 +3970,10 @@ xtrabackup_backup_func(void)
mutex_exit(&log_sys->mutex);
reread_log_header:
- fil_io(OS_FILE_READ | OS_FILE_LOG, TRUE, max_cp_group->space_id,
+ fil_io(OS_FILE_READ | OS_FILE_LOG, true, max_cp_group->space_id,
0,
0, 0, LOG_FILE_HDR_SIZE,
- log_hdr_buf, max_cp_group);
+ log_hdr_buf, max_cp_group, NULL);
/* check consistency of log file header to copy */
mutex_enter(&log_sys->mutex);
@@ -4427,7 +4447,7 @@ loop:
}
static void
-xtrabackup_stats_func(void)
+xtrabackup_stats_func(int argc, char **argv)
{
ulint n;
@@ -4439,7 +4459,7 @@ xtrabackup_stats_func(void)
exit(EXIT_FAILURE);
}
msg("xtrabackup: cd to %s\n", mysql_real_data_home);
-
+ encryption_plugin_prepare_init(argc, argv);
mysql_data_home= mysql_data_home_buff;
mysql_data_home[0]=FN_CURLIB; // all paths are relative from here
mysql_data_home[1]=0;
@@ -4654,7 +4674,7 @@ xtrabackup_init_temp_log(void)
ibool success;
ulint field;
- byte log_buf[UNIV_PAGE_SIZE_MAX * 128]; /* 2 MB */
+ byte* log_buf= (byte *)malloc(UNIV_PAGE_SIZE_MAX * 128); /* 2 MB */
ib_int64_t file_size;
@@ -4668,6 +4688,10 @@ xtrabackup_init_temp_log(void)
max_no = 0;
+ if (!log_buf) {
+ goto error;
+ }
+
if (!xb_init_log_block_size()) {
goto error;
}
@@ -4688,7 +4712,7 @@ retry:
src_file = os_file_create_simple_no_error_handling(0, src_path,
OS_FILE_OPEN,
OS_FILE_READ_WRITE,
- &success);
+ &success,0);
if (!success) {
/* The following call prints an error message */
os_file_get_last_error(TRUE);
@@ -4700,7 +4724,7 @@ retry:
src_file = os_file_create_simple_no_error_handling(0, dst_path,
OS_FILE_OPEN,
OS_FILE_READ_WRITE,
- &success);
+ &success,0);
if (!success) {
os_file_get_last_error(TRUE);
msg(" xtrabackup: Fatal error: cannot find %s.\n",
@@ -4798,7 +4822,7 @@ not_consistent:
mach_write_to_4(log_buf + LOG_CHECKPOINT_1
+ LOG_CHECKPOINT_OFFSET_LOW32,
LOG_FILE_HDR_SIZE +
- (max_lsn -
+ (ulint)(max_lsn -
ut_uint64_align_down(max_lsn,
OS_FILE_LOG_BLOCK_SIZE)));
mach_write_to_4(log_buf + LOG_CHECKPOINT_1
@@ -4815,7 +4839,7 @@ not_consistent:
mach_write_to_4(log_buf + LOG_CHECKPOINT_2
+ LOG_CHECKPOINT_OFFSET_LOW32,
LOG_FILE_HDR_SIZE +
- (max_lsn -
+ (ulint)(max_lsn -
ut_uint64_align_down(max_lsn,
OS_FILE_LOG_BLOCK_SIZE)));
mach_write_to_4(log_buf + LOG_CHECKPOINT_2
@@ -4916,15 +4940,17 @@ not_consistent:
goto error;
}
xtrabackup_logfile_is_renamed = TRUE;
-
+ free(log_buf);
return(FALSE);
skip_modify:
+ free(log_buf);
os_file_close(src_file);
src_file = XB_FILE_UNDEFINED;
return(FALSE);
error:
+ free(log_buf);
if (src_file != XB_FILE_UNDEFINED)
os_file_close(src_file);
msg("xtrabackup: Error: xtrabackup_init_temp_log() failed.\n");
@@ -4977,7 +5003,7 @@ xb_space_create_file(
*file = os_file_create_simple_no_error_handling(0, path, OS_FILE_CREATE,
OS_FILE_READ_WRITE,
- &ret);
+ &ret,0);
if (!ret) {
msg("xtrabackup: cannot create file %s\n", path);
return ret;
@@ -5171,7 +5197,7 @@ xb_delta_open_matching_space(
/* No matching space found. create the new one. */
if (!fil_space_create(dest_space_name, space_id, 0,
- FIL_TABLESPACE)) {
+ FIL_TABLESPACE, 0, false)) {
msg("xtrabackup: Cannot create tablespace %s\n",
dest_space_name);
goto exit;
@@ -5201,7 +5227,7 @@ found:
file = os_file_create_simple_no_error_handling(0, real_name,
OS_FILE_OPEN,
OS_FILE_READ_WRITE,
- &ok);
+ &ok,0);
if (ok) {
*success = TRUE;
@@ -5290,7 +5316,7 @@ xtrabackup_apply_delta(
src_file = os_file_create_simple_no_error_handling(0, src_path,
OS_FILE_OPEN,
OS_FILE_READ_WRITE,
- &success);
+ &success,0);
if (!success) {
os_file_get_last_error(TRUE);
msg("xtrabackup: error: cannot open %s\n", src_path);
@@ -5315,11 +5341,11 @@ xtrabackup_apply_delta(
/* allocate buffer for incremental backup (4096 pages) */
incremental_buffer_base = static_cast<byte *>
- (ut_malloc((UNIV_PAGE_SIZE_MAX / 4 + 1) *
- UNIV_PAGE_SIZE_MAX));
+ (ut_malloc((page_size / 4 + 1) *
+ page_size));
incremental_buffer = static_cast<byte *>
(ut_align(incremental_buffer_base,
- UNIV_PAGE_SIZE_MAX));
+ page_size));
msg("Applying %s to %s...\n", src_path, dst_path);
@@ -5637,7 +5663,7 @@ xtrabackup_close_temp_log(my_bool clear_flag)
src_file = os_file_create_simple_no_error_handling(0, src_path,
OS_FILE_OPEN,
OS_FILE_READ_WRITE,
- &success);
+ &success,0);
if (!success) {
goto error;
}
@@ -5701,7 +5727,7 @@ xb_export_cfg_write_index_fields(
}
/* Include the NUL byte in the length. */
- ib_uint32_t len = strlen(field->name) + 1;
+ ib_uint32_t len = (ib_uint32_t)strlen(field->name) + 1;
ut_a(len > 1);
mach_write_to_4(row, len);
@@ -5791,7 +5817,7 @@ xb_export_cfg_write_indexes(
/* Write the length of the index name.
NUL byte is included in the length. */
- ib_uint32_t len = strlen(index->name) + 1;
+ ib_uint32_t len = (ib_uint32_t)strlen(index->name) + 1;
ut_a(len > 1);
mach_write_to_4(row, len);
@@ -5865,7 +5891,7 @@ xb_export_cfg_write_table(
col_name = dict_table_get_col_name(table, dict_col_get_no(col));
/* Include the NUL byte in the length. */
- len = strlen(col_name) + 1;
+ len = (ib_uint32_t)strlen(col_name) + 1;
ut_a(len > 1);
mach_write_to_4(row, len);
@@ -5909,7 +5935,7 @@ xb_export_cfg_write_header(
const char* hostname = "Hostname unknown";
/* The server hostname includes the NUL byte. */
- len = strlen(hostname) + 1;
+ len = (ib_uint32_t)strlen(hostname) + 1;
mach_write_to_4(value, len);
if (fwrite(&value, 1, sizeof(value), file) != sizeof(value)
@@ -5922,7 +5948,7 @@ xb_export_cfg_write_header(
/* The table name includes the NUL byte. */
ut_a(table->name != 0);
- len = strlen(table->name) + 1;
+ len = (ib_uint32_t)strlen(table->name) + 1;
/* Write the table name. */
mach_write_to_4(value, len);
@@ -6122,7 +6148,7 @@ store_binlog_info(
}
static void
-xtrabackup_prepare_func(void)
+xtrabackup_prepare_func(int argc, char ** argv)
{
ulint err;
datafiles_iter_t *it;
@@ -6140,6 +6166,8 @@ xtrabackup_prepare_func(void)
}
msg("xtrabackup: cd to %s\n", xtrabackup_real_target_dir);
+ encryption_plugin_prepare_init(argc, argv);
+
xtrabackup_target_dir= mysql_data_home_buff;
xtrabackup_target_dir[0]=FN_CURLIB; // all paths are relative from here
xtrabackup_target_dir[1]=0;
@@ -6193,7 +6221,6 @@ skip_check:
/* Create logfiles for recovery from 'xtrabackup_logfile', before start InnoDB */
srv_max_n_threads = 1000;
- os_sync_mutex = NULL;
ut_mem_init();
/* temporally dummy value to avoid crash */
srv_page_size_shift = 14;
@@ -6204,6 +6231,11 @@ skip_check:
mem_init(srv_mem_pool_size);
ut_crc32_init();
+#ifdef WITH_INNODB_DISALLOW_WRITES
+ srv_allow_writes_event = os_event_create();
+ os_event_set(srv_allow_writes_event);
+#endif
+
xb_filters_init();
if(!innobase_log_arch_dir && xtrabackup_init_temp_log())
@@ -6213,25 +6245,6 @@ skip_check:
goto error_cleanup;
}
- /* Expand compacted datafiles */
-
- if (xtrabackup_compact) {
- srv_compact_backup = TRUE;
-
- if (!xb_expand_datafiles()) {
- goto error_cleanup;
- }
-
- /* Reset the 'compact' flag in xtrabackup_checkpoints so we
- don't expand on subsequent invocations. */
- xtrabackup_compact = FALSE;
- if (!xtrabackup_write_metadata(metadata_path)) {
- msg("xtrabackup: error: xtrabackup_write_metadata() "
- "failed\n");
- goto error_cleanup;
- }
- }
-
xb_normalize_init_values();
if (xtrabackup_incremental || innobase_log_arch_dir) {
@@ -6262,17 +6275,16 @@ skip_check:
xb_filter_hash_free(inc_dir_tables_hash);
}
- sync_close();
- sync_initialized = FALSE;
if (fil_system) {
fil_close();
}
- os_sync_free();
+
mem_close();
- os_sync_mutex = NULL;
ut_free_all_mem();
innodb_free_param();
+ sync_close();
+ sync_initialized = FALSE;
/* Reset the configuration as it might have been changed by
xb_data_files_init(). */
@@ -6280,8 +6292,7 @@ skip_check:
goto error_cleanup;
}
- srv_apply_log_only = (ibool) xtrabackup_apply_log_only;
- srv_rebuild_indexes = (ibool) xtrabackup_rebuild_indexes;
+ srv_apply_log_only = (bool) xtrabackup_apply_log_only;
/* increase IO threads */
if(srv_n_file_io_threads < 10) {
@@ -6291,7 +6302,7 @@ skip_check:
if (innobase_log_arch_dir) {
srv_arch_dir = innobase_log_arch_dir;
- srv_archive_recovery = TRUE;
+ srv_archive_recovery = true;
if (xtrabackup_archived_to_lsn) {
if (xtrabackup_archived_to_lsn < metadata_last_lsn) {
msg("xtrabackup: warning: logs applying lsn "
@@ -6317,7 +6328,7 @@ skip_check:
Unfinished transactions are not rolled back during log applying
as they can be finished at the firther files applyings.
*/
- srv_apply_log_only = xtrabackup_apply_log_only = TRUE;
+ srv_apply_log_only = xtrabackup_apply_log_only = true;
if (!xtrabackup_arch_search_files(min_flushed_lsn)) {
goto error_cleanup;
@@ -6340,6 +6351,12 @@ skip_check:
"xtrabackup: Using %lld bytes for buffer pool "
"(set by --use-memory parameter)\n", xtrabackup_use_memory);
+ srv_max_buf_pool_modified_pct = (double)max_buf_pool_modified_pct;
+
+ if (srv_max_dirty_pages_pct_lwm > srv_max_buf_pool_modified_pct) {
+ srv_max_dirty_pages_pct_lwm = srv_max_buf_pool_modified_pct;
+ }
+
if(innodb_init())
goto error_cleanup;
@@ -6423,15 +6440,20 @@ skip_check:
/* node exist == file exist, here */
strcpy(info_file_path, node->name);
+#ifdef _WIN32
+ for (int i = 0; info_file_path[i]; i++)
+ if (info_file_path[i] == '\\')
+ info_file_path[i]= '/';
+#endif
strcpy(info_file_path +
strlen(info_file_path) -
4, ".exp");
- len = strlen(info_file_path);
+ len =(ib_uint32_t)strlen(info_file_path);
p = info_file_path;
prev = NULL;
- while ((next = strchr(p, SRV_PATH_SEPARATOR)) != NULL)
+ while ((next = strchr(p, '/')) != NULL)
{
prev = p;
p = next + 1;
@@ -6503,7 +6525,7 @@ skip_check:
info_file_path,
OS_FILE_OVERWRITE,
OS_FILE_NORMAL, OS_DATA_FILE,
- &success);
+ &success,0);
if (!success) {
os_file_get_last_error(TRUE);
goto next_node;
@@ -6569,8 +6591,9 @@ next_node:
}
exit(EXIT_FAILURE);
}
-
+#ifdef WITH_WSREP
xb_write_galera_info(xtrabackup_incremental);
+#endif
if(innodb_end())
goto error_cleanup;
@@ -6578,7 +6601,6 @@ next_node:
innodb_free_param();
sync_initialized = FALSE;
- os_sync_mutex = NULL;
/* re-init necessary components */
ut_mem_init();
@@ -6630,9 +6652,7 @@ next_node:
if (fil_system) {
fil_close();
}
- os_sync_free();
- // mem_close();
- os_sync_mutex = NULL;
+
ut_free_all_mem();
/* start InnoDB once again to create log files */
@@ -6643,8 +6663,7 @@ next_node:
goto error;
}
- srv_apply_log_only = FALSE;
- srv_rebuild_indexes = FALSE;
+ srv_apply_log_only = false;
/* increase IO threads */
if(srv_n_file_io_threads < 10) {
@@ -6677,40 +6696,6 @@ error:
}
/**************************************************************************
-Signals-related setup. */
-static
-void
-setup_signals()
-/*===========*/
-{
- struct sigaction sa;
-
- /* Print a stacktrace on some signals */
- sa.sa_flags = SA_RESETHAND | SA_NODEFER;
- sigemptyset(&sa.sa_mask);
- sigprocmask(SIG_SETMASK,&sa.sa_mask,NULL);
-#ifdef HAVE_STACKTRACE
- my_init_stacktrace();
-#endif
- sa.sa_handler = handle_fatal_signal;
- sigaction(SIGSEGV, &sa, NULL);
- sigaction(SIGABRT, &sa, NULL);
- sigaction(SIGBUS, &sa, NULL);
- sigaction(SIGILL, &sa, NULL);
- sigaction(SIGFPE, &sa, NULL);
-
-#ifdef __linux__
- /* Ensure xtrabackup process is killed when the parent one
- (innobackupex) is terminated with an unhandled signal */
-
- if (prctl(PR_SET_PDEATHSIG, SIGKILL)) {
- msg("prctl() failed with errno = %d\n", errno);
- exit(EXIT_FAILURE);
- }
-#endif
-}
-
-/**************************************************************************
Append group name to xb_load_default_groups list. */
static
void
@@ -6778,11 +6763,6 @@ xb_init()
}
if (xtrabackup_backup) {
-
- if (!opt_noversioncheck) {
- version_check();
- }
-
if ((mysql_connection = xb_mysql_connect()) == NULL) {
return(false);
}
@@ -6791,6 +6771,7 @@ xb_init()
return(false);
}
+ encryption_plugin_backup_init(mysql_connection);
history_start_time = time(NULL);
}
@@ -6798,9 +6779,74 @@ xb_init()
return(true);
}
+
+extern void init_signals(void);
+
+#include <sql_locale.h>
+
+/* Messages . Avoid loading errmsg.sys file */
+void setup_error_messages()
+{
+ static const char *all_msgs[ER_ERROR_LAST - ER_ERROR_FIRST +1];
+ my_default_lc_messages = &my_locale_en_US;
+ my_default_lc_messages->errmsgs->errmsgs = all_msgs;
+
+ /* Populate the necessary error messages */
+ struct {
+ int id;
+ const char *fmt;
+ }
+ xb_msgs[] =
+ {
+ { ER_DATABASE_NAME,"Database" },
+ { ER_TABLE_NAME,"Table"},
+ { ER_PARTITION_NAME, "Partition" },
+ { ER_SUBPARTITION_NAME, "Subpartition" },
+ { ER_TEMPORARY_NAME, "Temporary"},
+ { ER_RENAMED_NAME, "Renamed"},
+ { ER_CANT_FIND_DL_ENTRY, "Can't find symbol '%-.128s' in library"},
+ { ER_CANT_OPEN_LIBRARY, "Can't open shared library '%-.192s' (errno: %d, %-.128s)" },
+ { ER_OUTOFMEMORY, "Out of memory; restart server and try again (needed %d bytes)" },
+ { ER_CANT_OPEN_LIBRARY, "Can't open shared library '%-.192s' (errno: %d, %-.128s)" },
+ { ER_UDF_NO_PATHS, "No paths allowed for shared library" },
+ { ER_CANT_INITIALIZE_UDF,"Can't initialize function '%-.192s'; %-.80s"},
+ { ER_PLUGIN_IS_NOT_LOADED,"Plugin '%-.192s' is not loaded" }
+ };
+
+ for (int i = 0; i < (int)array_elements(all_msgs); i++)
+ all_msgs[i] = "Unknown error";
+
+ for (int i = 0; i < (int)array_elements(xb_msgs); i++)
+ all_msgs[xb_msgs[i].id - ER_ERROR_FIRST] = xb_msgs[i].fmt;
+}
+
+extern my_bool(*dict_check_if_skip_table)(const char* name) ;
+
void
handle_options(int argc, char **argv, char ***argv_client, char ***argv_server)
{
+ /* Setup some variables for Innodb.*/
+
+ srv_xtrabackup = true;
+
+
+ files_charset_info = &my_charset_utf8_general_ci;
+ dict_check_if_skip_table = check_if_skip_table;
+
+ setup_error_messages();
+ sys_var_init();
+ plugin_mutex_init();
+ mysql_rwlock_init(key_rwlock_LOCK_system_variables_hash, &LOCK_system_variables_hash);
+ opt_stack_trace = 1;
+ test_flags |= TEST_SIGINT;
+ init_signals();
+#ifndef _WIN32
+ /* Exit process on SIGINT. */
+ my_sigset(SIGINT, SIG_DFL);
+#endif
+
+ sf_leaking_memory = 0; /* don't report memory leaks on early exist */
+
int i;
int ho_error;
@@ -6811,9 +6857,6 @@ handle_options(int argc, char **argv, char ***argv_client, char ***argv_server)
int argc_client = argc;
int argc_server = argc;
- *argv_client = argv;
- *argv_server = argv;
-
/* scan options for group and config file to load defaults from */
for (i = 1; i < argc; i++) {
@@ -6859,12 +6902,27 @@ handle_options(int argc, char **argv, char ***argv_client, char ***argv_server)
if (prepare && target_dir) {
snprintf(conf_file, sizeof(conf_file),
"%s/backup-my.cnf", target_dir);
+ if (!strncmp(argv[1], "--defaults-file=", 16)) {
+ /* Remove defaults-file*/
+ for (int i = 2; ; i++) {
+ if ((argv[i-1]= argv[i]) == 0)
+ break;
+ }
+ argc--;
+ }
}
+
+ *argv_client = argv;
+ *argv_server = argv;
if (load_defaults(conf_file, xb_server_default_groups,
&argc_server, argv_server)) {
exit(EXIT_FAILURE);
}
+ int n;
+ for (n = 0; (*argv_server)[n]; n++) {};
+ argc_server = n;
+
print_param_str <<
"# This MySQL options file was generated by XtraBackup.\n"
"[" << defaults_group << "]\n";
@@ -6879,6 +6937,7 @@ handle_options(int argc, char **argv, char ***argv_client, char ***argv_server)
optp->u_max_value = (G_PTR *) &global_max_value;
}
+
/* Throw a descriptive error if --defaults-file or --defaults-extra-file
is not the first command line argument */
for (int i = 2 ; i < argc ; i++) {
@@ -6913,6 +6972,9 @@ handle_options(int argc, char **argv, char ***argv_client, char ***argv_server)
exit(EXIT_FAILURE);
}
+ for (n = 0; (*argv_client)[n]; n++) {};
+ argc_client = n;
+
if (strcmp(base_name(my_progname), INNOBACKUPEX_BIN_NAME) == 0 &&
argc_client > 0) {
/* emulate innobackupex script */
@@ -6959,10 +7021,16 @@ int main(int argc, char **argv)
{
char **client_defaults, **server_defaults;
char cwd[FN_REFLEN];
- my_bool is_symdir;
- setup_signals();
+ if (argc > 1 && (strcmp(argv[1], "--innobackupex") == 0))
+ {
+ argv++;
+ argc--;
+ argv[0] = "innobackupex";
+ innobackupex_mode = true;
+ }
+ init_signals();
MY_INIT(argv[0]);
pthread_key_create(&THR_THD, NULL);
@@ -6977,11 +7045,18 @@ int main(int argc, char **argv)
exit(EXIT_FAILURE);
}
- system_charset_info= &my_charset_utf8_general_ci;
+ system_charset_info = &my_charset_utf8_general_ci;
key_map_full.set_all();
handle_options(argc, argv, &client_defaults, &server_defaults);
+ int argc_server;
+ for (argc_server = 0; server_defaults[argc_server]; argc_server++) {}
+
+ int argc_client;
+ for (argc_client = 0; client_defaults[argc_client]; argc_client++) {}
+
+
if (innobackupex_mode) {
if (!ibx_init()) {
exit(EXIT_FAILURE);
@@ -7002,14 +7077,14 @@ int main(int argc, char **argv)
my_load_path(xtrabackup_real_target_dir,
xtrabackup_target_dir, cwd);
unpack_dirname(xtrabackup_real_target_dir,
- xtrabackup_real_target_dir, &is_symdir);
+ xtrabackup_real_target_dir);
xtrabackup_target_dir= xtrabackup_real_target_dir;
if (xtrabackup_incremental_basedir) {
my_load_path(xtrabackup_real_incremental_basedir,
xtrabackup_incremental_basedir, cwd);
unpack_dirname(xtrabackup_real_incremental_basedir,
- xtrabackup_real_incremental_basedir, &is_symdir);
+ xtrabackup_real_incremental_basedir);
xtrabackup_incremental_basedir =
xtrabackup_real_incremental_basedir;
}
@@ -7018,7 +7093,7 @@ int main(int argc, char **argv)
my_load_path(xtrabackup_real_incremental_dir,
xtrabackup_incremental_dir, cwd);
unpack_dirname(xtrabackup_real_incremental_dir,
- xtrabackup_real_incremental_dir, &is_symdir);
+ xtrabackup_real_incremental_dir);
xtrabackup_incremental_dir = xtrabackup_real_incremental_dir;
}
@@ -7026,7 +7101,7 @@ int main(int argc, char **argv)
my_load_path(xtrabackup_real_extra_lsndir,
xtrabackup_extra_lsndir, cwd);
unpack_dirname(xtrabackup_real_extra_lsndir,
- xtrabackup_real_extra_lsndir, &is_symdir);
+ xtrabackup_real_extra_lsndir);
xtrabackup_extra_lsndir = xtrabackup_real_extra_lsndir;
}
@@ -7181,11 +7256,12 @@ int main(int argc, char **argv)
/* --stats */
if (xtrabackup_stats)
- xtrabackup_stats_func();
+ xtrabackup_stats_func(argc_server,server_defaults);
/* --prepare */
- if (xtrabackup_prepare)
- xtrabackup_prepare_func();
+ if (xtrabackup_prepare) {
+ xtrabackup_prepare_func(argc_server, server_defaults);
+ }
if (xtrabackup_copy_back || xtrabackup_move_back) {
if (!check_if_param_set("datadir")) {
@@ -7206,7 +7282,6 @@ int main(int argc, char **argv)
ibx_cleanup();
}
- xb_regex_end();
free_defaults(client_defaults);
free_defaults(server_defaults);
diff --git a/extra/mariabackup/xtrabackup.h b/extra/mariabackup/xtrabackup.h
index b31028175c9..de21da31741 100644
--- a/extra/mariabackup/xtrabackup.h
+++ b/extra/mariabackup/xtrabackup.h
@@ -63,8 +63,6 @@ extern lsn_t checkpoint_lsn_start;
extern xb_page_bitmap *changed_page_bitmap;
-extern ulint xtrabackup_rebuild_threads;
-
extern char *xtrabackup_incremental;
extern my_bool xtrabackup_incremental_force_scan;
@@ -80,7 +78,6 @@ extern char *xtrabackup_tables_file;
extern char *xtrabackup_databases;
extern char *xtrabackup_databases_file;
-extern my_bool xtrabackup_compact;
extern ibool xtrabackup_compress;
extern ibool xtrabackup_encrypt;
@@ -106,8 +103,14 @@ extern int xtrabackup_parallel;
extern my_bool xb_close_files;
extern const char *xtrabackup_compress_alg;
-extern uint xtrabackup_compress_threads;
-extern ulonglong xtrabackup_compress_chunk_size;
+#ifdef __cplusplus
+extern "C"{
+#endif
+ extern uint xtrabackup_compress_threads;
+ extern ulonglong xtrabackup_compress_chunk_size;
+#ifdef __cplusplus
+}
+#endif
extern ulong xtrabackup_encrypt_algo;
extern uint xtrabackup_encrypt_threads;
extern ulonglong xtrabackup_encrypt_chunk_size;
@@ -116,7 +119,6 @@ extern char *xtrabackup_incremental_basedir;
extern char *xtrabackup_extra_lsndir;
extern char *xtrabackup_incremental_dir;
extern ulint xtrabackup_log_copy_interval;
-extern my_bool xtrabackup_rebuild_indexes;
extern char *xtrabackup_stream_str;
extern long xtrabackup_throttle;
extern longlong xtrabackup_use_memory;
@@ -165,14 +167,6 @@ extern uint opt_safe_slave_backup_timeout;
extern const char *opt_history;
extern my_bool opt_decrypt;
-#if defined(HAVE_OPENSSL)
-extern my_bool opt_use_ssl;
-extern my_bool opt_ssl_verify_server_cert;
-#if !defined(HAVE_YASSL)
-extern char *opt_server_public_key;
-#endif
-#endif
-
enum binlog_info_enum { BINLOG_INFO_OFF, BINLOG_INFO_LOCKLESS, BINLOG_INFO_ON,
BINLOG_INFO_AUTO};
@@ -217,6 +211,14 @@ Check if parameter is set in defaults file or via command line argument
bool
check_if_param_set(const char *param);
+#if defined(HAVE_OPENSSL)
+extern my_bool opt_use_ssl;
+extern my_bool opt_ssl_verify_server_cert;
+#if !defined(HAVE_YASSL)
+extern char *opt_server_public_key;
+#endif
+#endif
+
void
xtrabackup_backup_func(void);
diff --git a/extra/mariabackup/xtrabackup_version.h.in b/extra/mariabackup/xtrabackup_version.h.in
deleted file mode 100644
index dc4c7992f8f..00000000000
--- a/extra/mariabackup/xtrabackup_version.h.in
+++ /dev/null
@@ -1,27 +0,0 @@
-/******************************************************
-Copyright (c) 2013 Percona LLC and/or its affiliates.
-
-Version numbers definitions.
-
-This program is free software; you can redistribute it and/or modify
-it under the terms of the GNU General Public License as published by
-the Free Software Foundation; version 2 of the License.
-
-This program is distributed in the hope that it will be useful,
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU General Public License for more details.
-
-You should have received a copy of the GNU General Public License
-along with this program; if not, write to the Free Software
-Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
-
-*******************************************************/
-
-#ifndef XB_VERSION_H
-#define XB_VERSION_H
-
-#define XTRABACKUP_VERSION "@XB_VERSION@"
-#define XTRABACKUP_REVISION "@XB_REVISION@"
-
-#endif /* XB_VERSION_H */
diff --git a/storage/xtradb/CMakeLists.txt b/storage/xtradb/CMakeLists.txt
index 8d1bd543d0c..77d8ca7fdbc 100644
--- a/storage/xtradb/CMakeLists.txt
+++ b/storage/xtradb/CMakeLists.txt
@@ -506,3 +506,6 @@ MYSQL_ADD_PLUGIN(xtradb ${INNOBASE_SOURCES} STORAGE_ENGINE
IF(TARGET xtradb AND NOT XTRADB_OK)
MESSAGE(FATAL_ERROR "Percona XtraDB is not supported on this platform")
ENDIF()
+
+ADD_SUBDIRECTORY(${CMAKE_SOURCE_DIR}/extra/mariabackup ${CMAKE_BINARY_DIR}/extra/mariabackup)
+