summaryrefslogtreecommitdiff
path: root/extra/innochecksum.cc
diff options
context:
space:
mode:
Diffstat (limited to 'extra/innochecksum.cc')
-rw-r--r--extra/innochecksum.cc2622
1 files changed, 1765 insertions, 857 deletions
diff --git a/extra/innochecksum.cc b/extra/innochecksum.cc
index 2536926513a..fa50ca72867 100644
--- a/extra/innochecksum.cc
+++ b/extra/innochecksum.cc
@@ -1,6 +1,6 @@
/*
- Copyright (c) 2005, 2012, Oracle and/or its affiliates.
- Copyright (c) 2014, 2015, MariaDB Corporation.
+ Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved.
+ Copyright (c) 2014, 2017, MariaDB Corporation.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
@@ -26,94 +26,137 @@
Published with a permission.
*/
+#include <my_config.h>
#include <my_global.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <sys/types.h>
#include <sys/stat.h>
-#ifndef __WIN__
+#ifdef HAVE_UNISTD_H
# include <unistd.h>
#endif
#include <my_getopt.h>
#include <m_string.h>
-#include <welcome_copyright_notice.h> /* ORACLE_WELCOME_COPYRIGHT_NOTICE */
+#include <welcome_copyright_notice.h> /* ORACLE_WELCOME_COPYRIGHT_NOTICE */
/* Only parts of these files are included from the InnoDB codebase.
The parts not included are excluded by #ifndef UNIV_INNOCHECKSUM. */
-#include "univ.i" /* include all of this */
-
-#define FLST_BASE_NODE_SIZE (4 + 2 * FIL_ADDR_SIZE)
-#define FLST_NODE_SIZE (2 * FIL_ADDR_SIZE)
-#define FSEG_PAGE_DATA FIL_PAGE_DATA
-#define MLOG_1BYTE (1)
-
-#include "ut0ut.h"
+#include "univ.i" /* include all of this */
+#include "page0size.h" /* page_size_t */
+#include "page0zip.h" /* page_zip_calc_checksum() */
+#include "page0page.h" /* PAGE_* */
+#include "trx0undo.h" /* TRX_UNDO_* */
+#include "fut0lst.h" /* FLST_NODE_SIZE */
+#include "buf0checksum.h" /* buf_calc_page_*() */
+#include "fil0fil.h" /* FIL_* */
+#include "fil0crypt.h"
+#include "os0file.h"
+#include "fsp0fsp.h" /* fsp_flags_get_page_size() &
+ fsp_flags_get_zip_size() */
+#include "mach0data.h" /* mach_read_from_4() */
+#include "ut0crc32.h" /* ut_crc32_init() */
+#include "fsp0pagecompress.h" /* fil_get_compression_alg_name */
#include "ut0byte.h"
#include "mach0data.h"
-#include "fsp0types.h"
-#include "rem0rec.h"
-#include "buf0checksum.h" /* buf_calc_page_*() */
-#include "fil0fil.h" /* FIL_* */
-#include "page0page.h" /* PAGE_* */
-#include "page0zip.h" /* page_zip_*() */
-#include "trx0undo.h" /* TRX_* */
-#include "fsp0fsp.h" /* fsp_flags_get_page_size() &
- fsp_flags_get_zip_size() */
-#include "ut0crc32.h" /* ut_crc32_init() */
-#include "fsp0pagecompress.h" /* fil_get_compression_alg_name */
-#ifdef UNIV_NONINL
-# include "fsp0fsp.ic"
-# include "mach0data.ic"
-# include "ut0rnd.ic"
+#ifndef PRIuMAX
+#define PRIuMAX "llu"
#endif
/* Global variables */
-static my_bool verbose;
-static my_bool debug;
-static my_bool skip_corrupt;
-static my_bool just_count;
-static ulong start_page;
-static ulong end_page;
-static ulong do_page;
-static my_bool use_end_page;
-static my_bool do_one_page;
-static my_bool per_page_details;
+static bool verbose;
+static bool just_count;
+static uintmax_t start_page;
+static uintmax_t end_page;
+static uintmax_t do_page;
+static bool use_end_page;
+static bool do_one_page;
+/* replaces declaration in srv0srv.c */
+ulong srv_page_size;
+page_size_t univ_page_size(0, 0, false);
+extern ulong srv_checksum_algorithm;
+/* Current page number (0 based). */
+uintmax_t cur_page_num;
+/* Skip the checksum verification. */
+static bool no_check;
+/* Enabled for strict checksum verification. */
+bool strict_verify = 0;
+/* Enabled for rewrite checksum. */
+static bool do_write;
+/* Mismatches count allowed (0 by default). */
+static uintmax_t allow_mismatches;
+static bool page_type_summary;
+static bool page_type_dump;
+/* Store filename for page-type-dump option. */
+char* page_dump_filename = 0;
+/* skip the checksum verification & rewrite if page is doublewrite buffer. */
+static bool skip_page = 0;
+const char *dbug_setting = "FALSE";
+char* log_filename = NULL;
+/* User defined filename for logging. */
+FILE* log_file = NULL;
+/* Enabled for log write option. */
+static bool is_log_enabled = false;
static my_bool do_leaf;
static ulong n_merge;
-ulong srv_page_size; /* replaces declaration in srv0srv.c */
-static ulong physical_page_size; /* Page size in bytes on disk. */
-static ulong logical_page_size; /* Page size when uncompressed. */
-static bool compressed= false; /* Is tablespace compressed */
-
-int n_undo_state_active;
-int n_undo_state_cached;
-int n_undo_state_to_free;
-int n_undo_state_to_purge;
-int n_undo_state_prepared;
-int n_undo_state_other;
-int n_undo_insert, n_undo_update, n_undo_other;
-int n_bad_checksum;
-int n_fil_page_index;
-int n_fil_page_undo_log;
-int n_fil_page_inode;
-int n_fil_page_ibuf_free_list;
-int n_fil_page_allocated;
-int n_fil_page_ibuf_bitmap;
-int n_fil_page_type_sys;
-int n_fil_page_type_trx_sys;
-int n_fil_page_type_fsp_hdr;
-int n_fil_page_type_allocated;
-int n_fil_page_type_xdes;
-int n_fil_page_type_blob;
-int n_fil_page_type_zblob;
-int n_fil_page_type_other;
-int n_fil_page_type_page_compressed;
-int n_fil_page_type_page_compressed_encrypted;
-
-int n_fil_page_max_index_id;
+#ifndef _WIN32
+/* advisory lock for non-window system. */
+struct flock lk;
+#endif /* _WIN32 */
+
+/* Strict check algorithm name. */
+static ulong strict_check;
+/* Rewrite checksum algorithm name. */
+static ulong write_check;
+
+/* Innodb page type. */
+struct innodb_page_type {
+ int n_undo_state_active;
+ int n_undo_state_cached;
+ int n_undo_state_to_free;
+ int n_undo_state_to_purge;
+ int n_undo_state_prepared;
+ int n_undo_state_other;
+ int n_undo_insert;
+ int n_undo_update;
+ int n_undo_other;
+ int n_fil_page_index;
+ int n_fil_page_undo_log;
+ int n_fil_page_inode;
+ int n_fil_page_ibuf_free_list;
+ int n_fil_page_ibuf_bitmap;
+ int n_fil_page_type_sys;
+ int n_fil_page_type_trx_sys;
+ int n_fil_page_type_fsp_hdr;
+ int n_fil_page_type_allocated;
+ int n_fil_page_type_xdes;
+ int n_fil_page_type_blob;
+ int n_fil_page_type_zblob;
+ int n_fil_page_type_other;
+ int n_fil_page_type_zblob2;
+ int n_fil_page_type_page_compressed;
+ int n_fil_page_type_page_compressed_encrypted;
+} page_type;
+
+/* Possible values for "--strict-check" for strictly verify checksum
+and "--write" for rewrite checksum. */
+static const char *innochecksum_algorithms[] = {
+ "crc32",
+ "crc32",
+ "innodb",
+ "innodb",
+ "none",
+ "none",
+ NullS
+};
+
+/* Used to define an enumerate type of the "innochecksum algorithm". */
+static TYPELIB innochecksum_algorithms_typelib = {
+ array_elements(innochecksum_algorithms)-1,"",
+ innochecksum_algorithms, NULL
+};
#define SIZE_RANGES_FOR_PAGE 10
#define NUM_RETRIES 3
@@ -143,7 +186,7 @@ struct per_index_stats {
last element for pages with more than logical_page_size */
unsigned long long pages_in_size_range[SIZE_RANGES_FOR_PAGE+2];
- std::map<ulint, per_page_stats> leaves;
+ std::map<unsigned long long, per_page_stats> leaves;
per_index_stats():pages(0), leaf_pages(0), first_leaf_page(0),
count(0), free_pages(0), max_data_size(0), total_n_recs(0),
@@ -157,185 +200,622 @@ std::map<unsigned long long, per_index_stats> index_ids;
bool encrypted = false;
-/* Get the page size of the filespace from the filespace header. */
+ulint
+page_is_comp(
+/*=========*/
+ const page_t* page) /*!< in: index page */
+{
+ return(page_header_get_field(page, PAGE_N_HEAP) & 0x8000);
+}
+
+bool
+page_is_leaf(
+/*=========*/
+ const page_t* page) /*!< in: page */
+{
+ return(!*(const uint16*) (page + (PAGE_HEADER + PAGE_LEVEL)));
+}
+
+ulint
+page_get_page_no(
+/*=============*/
+ const page_t* page) /*!< in: page */
+{
+ return(mach_read_from_4(page + FIL_PAGE_OFFSET));
+}
+#define FSEG_HEADER_SIZE 10 /*!< Length of the file system
+ header, in bytes */
+#define REC_N_NEW_EXTRA_BYTES 5
+#define REC_N_OLD_EXTRA_BYTES 6
+#define PAGE_DATA (PAGE_HEADER + 36 + 2 * FSEG_HEADER_SIZE)
+ /* start of data on the page */
+#define PAGE_NEW_SUPREMUM (PAGE_DATA + 2 * REC_N_NEW_EXTRA_BYTES + 8)
+ /* offset of the page supremum record on a
+ new-style compact page */
+#define PAGE_NEW_SUPREMUM_END (PAGE_NEW_SUPREMUM + 8)
+#define PAGE_OLD_SUPREMUM (PAGE_DATA + 2 + 2 * REC_N_OLD_EXTRA_BYTES + 8)
+ /* offset of the page supremum record on an
+ old-style page */
+#define PAGE_OLD_SUPREMUM_END (PAGE_OLD_SUPREMUM + 9)
+ /* offset of the page supremum record end on
+ an old-style page */
+#define FLST_BASE_NODE_SIZE (4 + 2 * 6)
+#define XDES_ARR_OFFSET (FSP_HEADER_OFFSET + FSP_HEADER_SIZE)
+#define XDES_FREE_BIT 0
+#define XDES_BITMAP (FLST_NODE_SIZE + 12)
+#define XDES_BITS_PER_PAGE 2
+#define UT_BITS_IN_BYTES(b) (((b) + 7) / 8)
+#define XDES_SIZE \
+ (XDES_BITMAP \
+ + UT_BITS_IN_BYTES(FSP_EXTENT_SIZE * XDES_BITS_PER_PAGE))
+
+ulint
+page_get_data_size(
+/*===============*/
+ const page_t* page) /*!< in: index page */
+{
+ ulint ret;
+
+ ret = (ulint)(page_header_get_field(page, PAGE_HEAP_TOP)
+ - (page_is_comp(page)
+ ? PAGE_NEW_SUPREMUM_END
+ : PAGE_OLD_SUPREMUM_END)
+ - page_header_get_field(page, PAGE_GARBAGE));
+
+ ut_ad(ret < UNIV_PAGE_SIZE);
+
+ return(ret);
+}
+
+void print_index_leaf_stats(
+ unsigned long long id,
+ const per_index_stats& index,
+ FILE* fil_out)
+
+{
+ ulint page_no = index.first_leaf_page;
+ std::map<unsigned long long, per_page_stats>::const_iterator it_page = index.leaves.find(page_no);
+ fprintf(fil_out, "\nindex: %llu leaf page stats: n_pages = %llu\n",
+ id, index.leaf_pages);
+ fprintf(fil_out, "page_no\tdata_size\tn_recs\n");
+ while (it_page != index.leaves.end()) {
+ const per_page_stats& stat = it_page->second;
+ fprintf(fil_out, "%llu\t%lu\t%lu\n", it_page->first, stat.data_size, stat.n_recs);
+ page_no = stat.right_page_no;
+ it_page = index.leaves.find(page_no);
+ }
+}
+
+void defrag_analysis(
+ unsigned long long id,
+ const per_index_stats& index,
+ FILE* fil_out)
+{
+ // TODO: make it work for compressed pages too
+ std::map<unsigned long long, per_page_stats>::const_iterator it = index.leaves.find(index.first_leaf_page);
+ ulint n_pages = 0;
+ ulint n_leaf_pages = 0;
+ while (it != index.leaves.end()) {
+ ulint data_size_total = 0;
+ for (ulong i = 0; i < n_merge; i++) {
+ const per_page_stats& stat = it->second;
+ n_leaf_pages ++;
+ data_size_total += stat.data_size;
+ it = index.leaves.find(stat.right_page_no);
+ if (it == index.leaves.end()) {
+ break;
+ }
+ }
+ if (index.max_data_size) {
+ n_pages += data_size_total / index.max_data_size;
+ if (data_size_total % index.max_data_size != 0) {
+ n_pages += 1;
+ }
+ }
+ }
+
+ if (index.leaf_pages) {
+ fprintf(fil_out, "count = %lu free = %lu\n", index.count, index.free_pages);
+ }
+
+ fprintf(fil_out, "%llu\t\t%llu\t\t%lu\t\t%lu\t\t%lu\t\t%.2f\t%lu\n",
+ id, index.leaf_pages, n_leaf_pages, n_merge, n_pages,
+ 1.0 - (double)n_pages / (double)n_leaf_pages, index.max_data_size);
+}
+
+void print_leaf_stats(
+ FILE* fil_out)
+{
+ fprintf(fil_out, "\n**************************************************\n");
+ fprintf(fil_out, "index_id\t#leaf_pages\t#actual_leaf_pages\tn_merge\t"
+ "#leaf_after_merge\tdefrag\n");
+ for (std::map<unsigned long long, per_index_stats>::const_iterator it = index_ids.begin();
+ it != index_ids.end(); it++) {
+ const per_index_stats& index = it->second;
+
+ if (verbose) {
+ print_index_leaf_stats(it->first, index, fil_out);
+ }
+
+ if (n_merge) {
+ defrag_analysis(it->first, index, fil_out);
+ }
+ }
+}
+
+/** Get the page size of the filespace from the filespace header.
+@param[in] buf buffer used to read the page.
+@return page size */
static
-my_bool
+const page_size_t
get_page_size(
-/*==========*/
- FILE* f, /*!< in: file pointer, must be open
- and set to start of file */
- byte* buf, /*!< in: buffer used to read the page */
- ulong* logical_page_size, /*!< out: Logical/Uncompressed page size */
- ulong* physical_page_size) /*!< out: Physical/Commpressed page size */
+ byte* buf)
{
- ulong flags;
+ const ulint flags = mach_read_from_4(buf + FIL_PAGE_DATA
+ + FSP_SPACE_FLAGS);
- int bytes= fread(buf, 1, UNIV_PAGE_SIZE_MIN, f);
+ const ulint ssize = FSP_FLAGS_GET_PAGE_SSIZE(flags);
- if (ferror(f))
- {
- perror("Error reading file header");
- return FALSE;
- }
+ if (ssize == 0) {
+ srv_page_size = UNIV_PAGE_SIZE_ORIG;
+ } else {
+ srv_page_size = ((UNIV_ZIP_SIZE_MIN >> 1) << ssize);
+ }
- if (bytes != UNIV_PAGE_SIZE_MIN)
- {
- fprintf(stderr, "Error; Was not able to read the minimum page size ");
- fprintf(stderr, "of %d bytes. Bytes read was %d\n", UNIV_PAGE_SIZE_MIN, bytes);
- return FALSE;
- }
+ univ_page_size.copy_from(
+ page_size_t(srv_page_size, srv_page_size, false));
- rewind(f);
+ return(page_size_t(flags));
+}
- flags = mach_read_from_4(buf + FIL_PAGE_DATA + FSP_SPACE_FLAGS);
+#ifdef MYSQL_COMPRESSION
+/** Decompress a page
+@param[in,out] buf Page read from disk, uncompressed data will
+ also be copied to this page
+@param[in, out] scratch Page to use for temporary decompress
+@param[in] page_size scratch physical size
+@return true if decompress succeeded */
+static
+bool page_decompress(
+ byte* buf,
+ byte* scratch,
+ page_size_t page_size)
+{
+ dberr_t err;
- /* srv_page_size is used by InnoDB code as UNIV_PAGE_SIZE */
- srv_page_size = *logical_page_size = fsp_flags_get_page_size(flags);
+ /* Set the dblwr recover flag to false. */
+ err = os_file_decompress_page(
+ false, buf, scratch, page_size.physical());
- /* fsp_flags_get_zip_size() will return zero if not compressed. */
- *physical_page_size = fsp_flags_get_zip_size(flags);
- if (*physical_page_size == 0)
- {
- *physical_page_size= *logical_page_size;
- }
- else
- {
- compressed= true;
- }
+ return(err == DB_SUCCESS);
+}
+#endif
+#ifdef _WIN32
+/***********************************************//*
+ @param [in] error error no. from the getLastError().
- return TRUE;
-}
+ @retval error message corresponding to error no.
+*/
+static
+char*
+error_message(
+ int error)
+{
+ static char err_msg[1024] = {'\0'};
+ FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM,
+ NULL, error, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
+ (LPTSTR)err_msg, sizeof(err_msg), NULL );
+ return (err_msg);
+}
+#endif /* _WIN32 */
-/* command line argument to do page checks (that's it) */
-/* another argument to specify page ranges... seek to right spot and go from there */
+/***********************************************//*
+ @param>>_______[in] name>_____name of file.
+ @retval file pointer; file pointer is NULL when error occured.
+*/
-static struct my_option innochecksum_options[] =
+FILE*
+open_file(
+ const char* name)
{
- {"help", '?', "Displays this help and exits.",
- 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"info", 'I', "Synonym for --help.",
- 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"version", 'V', "Displays version information and exits.",
- 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"verbose", 'v', "Verbose (prints progress every 5 seconds).",
- &verbose, &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"debug", 'd', "Debug mode (prints checksums for each page, implies verbose).",
- &debug, &debug, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"skip_corrupt", 'u', "Skip corrupt pages.",
- &skip_corrupt, &skip_corrupt, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"count", 'c', "Print the count of pages in the file.",
- &just_count, &just_count, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"start_page", 's', "Start on this page number (0 based).",
- &start_page, &start_page, 0, GET_ULONG, REQUIRED_ARG,
- 0, 0, (longlong) 2L*1024L*1024L*1024L, 0, 1, 0},
- {"end_page", 'e', "End at this page number (0 based).",
- &end_page, &end_page, 0, GET_ULONG, REQUIRED_ARG,
- 0, 0, (longlong) 2L*1024L*1024L*1024L, 0, 1, 0},
- {"page", 'p', "Check only this page (0 based).",
- &do_page, &do_page, 0, GET_ULONG, REQUIRED_ARG,
- 0, 0, (longlong) 2L*1024L*1024L*1024L, 0, 1, 0},
- {"per_page_details", 'i', "Print out per-page detail information.",
- &per_page_details, &per_page_details, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}
- ,
- {"leaf", 'l', "Examine leaf index pages",
- &do_leaf, &do_leaf, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
- {"merge", 'm', "leaf page count if merge given number of consecutive pages",
- &n_merge, &n_merge, 0, GET_ULONG, REQUIRED_ARG,
- 0, 0, (longlong)10L, 0, 1, 0},
- {0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}
-};
+ int fd; /* file descriptor. */
+ FILE* fil_in;
+#ifdef _WIN32
+ HANDLE hFile; /* handle to open file. */
+ DWORD access; /* define access control */
+ int flags = 0; /* define the mode for file
+ descriptor */
+
+ if (do_write) {
+ access = GENERIC_READ | GENERIC_WRITE;
+ flags = _O_RDWR | _O_BINARY;
+ } else {
+ access = GENERIC_READ;
+ flags = _O_RDONLY | _O_BINARY;
+ }
+ /* CreateFile() also provide advisory lock with the usage of
+ access and share mode of the file.*/
+ hFile = CreateFile(
+ (LPCTSTR) name, access, 0L, NULL,
+ OPEN_EXISTING, NULL, NULL);
+
+ if (hFile == INVALID_HANDLE_VALUE) {
+ /* print the error message. */
+ fprintf(stderr, "Filename::%s %s\n", name,
+ error_message(GetLastError()));
+
+ return (NULL);
+ }
+
+ /* get the file descriptor. */
+ fd= _open_osfhandle((intptr_t)hFile, flags);
+#else /* _WIN32 */
+
+ int create_flag;
+ /* define the advisory lock and open file mode. */
+ if (do_write) {
+ create_flag = O_RDWR;
+ lk.l_type = F_WRLCK;
+ }
+ else {
+ create_flag = O_RDONLY;
+ lk.l_type = F_RDLCK;
+ }
-static void print_version(void)
+ fd = open(name, create_flag);
+
+ lk.l_whence = SEEK_SET;
+ lk.l_start = lk.l_len = 0;
+
+ if (fcntl(fd, F_SETLK, &lk) == -1) {
+ fprintf(stderr, "Error: Unable to lock file::"
+ " %s\n", name);
+ perror("fcntl");
+ return (NULL);
+ }
+#endif /* _WIN32 */
+
+ if (do_write) {
+ fil_in = fdopen(fd, "rb+");
+ } else {
+ fil_in = fdopen(fd, "rb");
+ }
+
+ return (fil_in);
+}
+
+/************************************************************//*
+ Read the content of file
+
+ @param [in,out] buf read the file in buffer
+ @param [in] partial_page_read enable when to read the
+ remaining buffer for first page.
+ @param [in] physical_page_size Physical/Commpressed page size.
+ @param [in,out] fil_in file pointer created for the
+ tablespace.
+ @retval no. of bytes read.
+*/
+ulong read_file(
+ byte* buf,
+ bool partial_page_read,
+ ulong physical_page_size,
+ FILE* fil_in)
{
- printf("%s Ver %s, for %s (%s)\n",
- my_progname, INNODB_VERSION_STR,
- SYSTEM_TYPE, MACHINE_TYPE);
+ ulong bytes = 0;
+
+ DBUG_ASSERT(physical_page_size >= UNIV_ZIP_SIZE_MIN);
+
+ if (partial_page_read) {
+ buf += UNIV_ZIP_SIZE_MIN;
+ physical_page_size -= UNIV_ZIP_SIZE_MIN;
+ bytes = UNIV_ZIP_SIZE_MIN;
+ }
+
+ bytes += ulong(fread(buf, 1, physical_page_size, fil_in));
+
+ return bytes;
}
-static void usage(void)
+/** Check if page is corrupted or not.
+@param[in] buf page frame
+@param[in] page_size page size
+@retval true if page is corrupted otherwise false. */
+static
+bool
+is_page_corrupted(
+ const byte* buf,
+ const page_size_t& page_size)
{
- print_version();
- puts(ORACLE_WELCOME_COPYRIGHT_NOTICE("2000"));
- printf("InnoDB offline file checksum utility.\n");
- printf("Usage: %s [-c] [-s <start page>] [-e <end page>] [-p <page>] [-v] [-d] <filename>\n", my_progname);
- my_print_help(innochecksum_options);
- my_print_variables(innochecksum_options);
+
+ /* enable if page is corrupted. */
+ bool is_corrupted;
+ /* use to store LSN values. */
+ ulint logseq;
+ ulint logseqfield;
+
+ if (!page_size.is_compressed()) {
+ /* check the stored log sequence numbers
+ for uncompressed tablespace. */
+ logseq = mach_read_from_4(buf + FIL_PAGE_LSN + 4);
+ logseqfield = mach_read_from_4(
+ buf + page_size.logical() -
+ FIL_PAGE_END_LSN_OLD_CHKSUM + 4);
+
+ if (is_log_enabled) {
+ fprintf(log_file,
+ "page::%" PRIuMAX
+ "; log sequence number:first = " ULINTPF
+ "; second = " ULINTPF "\n",
+ cur_page_num, logseq, logseqfield);
+ if (logseq != logseqfield) {
+ fprintf(log_file,
+ "Fail; page %" PRIuMAX
+ " invalid (fails log "
+ "sequence number check)\n",
+ cur_page_num);
+ }
+ }
+ }
+
+ /* FIXME: Read the page number from the tablespace header,
+ and check that every page carries the same page number. */
+
+ /* If page is encrypted, use different checksum calculation
+ as innochecksum can't decrypt pages. Note that some old InnoDB
+ versions did not initialize FIL_PAGE_FILE_FLUSH_LSN field
+ so if crypt checksum does not match we verify checksum using
+ normal method.
+ */
+ if (mach_read_from_4(buf+FIL_PAGE_FILE_FLUSH_LSN_OR_KEY_VERSION) != 0) {
+ is_corrupted = fil_space_verify_crypt_checksum(
+ const_cast<byte*>(buf), page_size,
+ strict_verify, is_log_enabled ? log_file : NULL,
+ mach_read_from_4(buf
+ + FIL_PAGE_ARCH_LOG_NO_OR_SPACE_ID),
+ cur_page_num);
+ } else {
+ is_corrupted = true;
+ }
+
+ if (is_corrupted) {
+ is_corrupted = buf_page_is_corrupted(
+ true, buf, page_size, NULL,
+ cur_page_num, strict_verify,
+ is_log_enabled, log_file);
+ }
+
+ return(is_corrupted);
}
-extern "C" my_bool
-innochecksum_get_one_option(
-/*========================*/
- int optid,
- const struct my_option *opt __attribute__((unused)),
- char *argument __attribute__((unused)))
+/********************************************//*
+ Check if page is doublewrite buffer or not.
+ @param [in] page buffer page
+
+ @retval true if page is doublewrite buffer otherwise false.
+*/
+static
+bool
+is_page_doublewritebuffer(
+ const byte* page)
{
- switch (optid) {
- case 'd':
- verbose=1; /* debug implies verbose... */
- break;
- case 'e':
- use_end_page= 1;
- break;
- case 'p':
- end_page= start_page= do_page;
- use_end_page= 1;
- do_one_page= 1;
- break;
- case 'V':
- print_version();
- exit(0);
- break;
- case 'I':
- case '?':
- usage();
- exit(0);
- break;
- }
- return 0;
+ if ((cur_page_num >= FSP_EXTENT_SIZE)
+ && (cur_page_num < FSP_EXTENT_SIZE * 3)) {
+ /* page is doublewrite buffer. */
+ return (true);
+ }
+
+ return (false);
}
-static int get_options(
-/*===================*/
- int *argc,
- char ***argv)
+/*******************************************************//*
+Check if page is empty or not.
+ @param [in] page page to checked for empty.
+ @param [in] len size of page.
+
+ @retval true if page is empty.
+ @retval false if page is not empty.
+*/
+static
+bool
+is_page_empty(
+ const byte* page,
+ size_t len)
{
- int ho_error;
+ while (len--) {
+ if (*page++) {
+ return (false);
+ }
+ }
+ return (true);
+}
- if ((ho_error=handle_options(argc, argv, innochecksum_options, innochecksum_get_one_option)))
- exit(ho_error);
+/********************************************************************//**
+Rewrite the checksum for the page.
+@param [in/out] page page buffer
+@param [in] physical_page_size page size in bytes on disk.
+@param [in] iscompressed Is compressed/Uncompressed Page.
- /* The next arg must be the filename */
- if (!*argc)
- {
- usage();
- return 1;
- }
- return 0;
-} /* get_options */
+@retval true : do rewrite
+@retval false : skip the rewrite as checksum stored match with
+ calculated or page is doublwrite buffer.
+*/
-/*********************************************************************//**
-Gets the file page type.
-@return type; NOTE that if the type has not been written to page, the
-return value not defined */
-ulint
-fil_page_get_type(
-/*==============*/
- uchar* page) /*!< in: file page */
+bool
+update_checksum(
+ byte* page,
+ ulong physical_page_size,
+ bool iscompressed)
{
- return(mach_read_from_2(page + FIL_PAGE_TYPE));
+ ib_uint32_t checksum = 0;
+ byte stored1[4]; /* get FIL_PAGE_SPACE_OR_CHKSUM field checksum */
+ byte stored2[4]; /* get FIL_PAGE_END_LSN_OLD_CHKSUM field checksum */
+
+ ut_ad(page);
+ /* If page is doublewrite buffer, skip the rewrite of checksum. */
+ if (skip_page) {
+ return (false);
+ }
+
+ memcpy(stored1, page + FIL_PAGE_SPACE_OR_CHKSUM, 4);
+ memcpy(stored2, page + physical_page_size -
+ FIL_PAGE_END_LSN_OLD_CHKSUM, 4);
+
+ /* Check if page is empty, exclude the checksum field */
+ if (is_page_empty(page + 4, physical_page_size - 12)
+ && is_page_empty(page + physical_page_size - 4, 4)) {
+
+ memset(page + FIL_PAGE_SPACE_OR_CHKSUM, 0, 4);
+ memset(page + physical_page_size -
+ FIL_PAGE_END_LSN_OLD_CHKSUM, 0, 4);
+
+ goto func_exit;
+ }
+
+ if (iscompressed) {
+ /* page is compressed */
+ checksum = page_zip_calc_checksum(
+ page, physical_page_size,
+ static_cast<srv_checksum_algorithm_t>(write_check));
+
+ mach_write_to_4(page + FIL_PAGE_SPACE_OR_CHKSUM, checksum);
+ if (is_log_enabled) {
+ fprintf(log_file, "page::%" PRIuMAX "; Updated checksum ="
+ " %u\n", cur_page_num, checksum);
+ }
+
+ } else {
+ /* page is uncompressed. */
+
+ /* Store the new formula checksum */
+ switch ((srv_checksum_algorithm_t) write_check) {
+
+ case SRV_CHECKSUM_ALGORITHM_CRC32:
+ case SRV_CHECKSUM_ALGORITHM_STRICT_CRC32:
+ checksum = buf_calc_page_crc32(page);
+ break;
+
+ case SRV_CHECKSUM_ALGORITHM_INNODB:
+ case SRV_CHECKSUM_ALGORITHM_STRICT_INNODB:
+ checksum = (ib_uint32_t)
+ buf_calc_page_new_checksum(page);
+ break;
+
+ case SRV_CHECKSUM_ALGORITHM_NONE:
+ case SRV_CHECKSUM_ALGORITHM_STRICT_NONE:
+ checksum = BUF_NO_CHECKSUM_MAGIC;
+ break;
+ /* no default so the compiler will emit a warning if new
+ enum is added and not handled here */
+ }
+
+ mach_write_to_4(page + FIL_PAGE_SPACE_OR_CHKSUM, checksum);
+ if (is_log_enabled) {
+ fprintf(log_file, "page::%" PRIuMAX "; Updated checksum field1"
+ " = %u\n", cur_page_num, checksum);
+ }
+
+ if (write_check == SRV_CHECKSUM_ALGORITHM_STRICT_INNODB
+ || write_check == SRV_CHECKSUM_ALGORITHM_INNODB) {
+ checksum = (ib_uint32_t)
+ buf_calc_page_old_checksum(page);
+ }
+
+ mach_write_to_4(page + physical_page_size -
+ FIL_PAGE_END_LSN_OLD_CHKSUM,checksum);
+
+ if (is_log_enabled) {
+ fprintf(log_file, "page::%" PRIuMAX "; Updated checksum "
+ "field2 = %u\n", cur_page_num, checksum);
+ }
+
+ }
+
+func_exit:
+ /* The following code is to check the stored checksum with the
+ calculated checksum. If it matches, then return FALSE to skip
+ the rewrite of checksum, otherwise return TRUE. */
+ if (iscompressed) {
+ if (!memcmp(stored1, page + FIL_PAGE_SPACE_OR_CHKSUM, 4)) {
+ return (false);
+ }
+ return (true);
+ }
+
+ if (!memcmp(stored1, page + FIL_PAGE_SPACE_OR_CHKSUM, 4)
+ && !memcmp(stored2, page + physical_page_size -
+ FIL_PAGE_END_LSN_OLD_CHKSUM, 4)) {
+ return (false);
+
+ }
+
+ return (true);
}
-/**************************************************************//**
-Gets the index id field of a page.
-@return index id */
-ib_uint64_t
-btr_page_get_index_id(
-/*==================*/
- uchar* page) /*!< in: index page */
+/**
+ Write the content to the file
+@param[in] filename name of the file.
+@param[in,out] file file pointer where content
+ have to be written
+@param[in] buf file buffer read
+@param[in] compressed Enabled if tablespace is
+ compressed.
+@param[in,out] pos current file position.
+@param[in] page_size page size in bytes on disk.
+
+@retval true if successfully written
+@retval false if a non-recoverable error occurred
+*/
+static
+bool
+write_file(
+ const char* filename,
+ FILE* file,
+ byte* buf,
+ bool compressed,
+ fpos_t* pos,
+ ulong page_size)
{
- return(mach_read_from_8(page + PAGE_HEADER + PAGE_INDEX_ID));
+ bool do_update;
+
+ do_update = update_checksum(buf, page_size, compressed);
+
+ if (file != stdin) {
+ if (do_update) {
+ /* Set the previous file pointer position
+ saved in pos to current file position. */
+ if (0 != fsetpos(file, pos)) {
+ perror("fsetpos");
+ return(false);
+ }
+ } else {
+ /* Store the current file position in pos */
+ if (0 != fgetpos(file, pos)) {
+ perror("fgetpos");
+ return(false);
+ }
+ return(true);
+ }
+ }
+
+ if (page_size
+ != fwrite(buf, 1, page_size, file == stdin ? stdout : file)) {
+ fprintf(stderr, "Failed to write page %" PRIuMAX " to %s: %s\n",
+ cur_page_num, filename, strerror(errno));
+
+ return(false);
+ }
+ if (file != stdin) {
+ fflush(file);
+ /* Store the current file position in pos */
+ if (0 != fgetpos(file, pos)) {
+ perror("fgetpos");
+ return(false);
+ }
+ }
+
+ return(true);
}
/********************************************************//**
@@ -360,694 +840,1122 @@ btr_page_get_prev(
return(mach_read_from_4(page + FIL_PAGE_PREV));
}
+ulint
+mach_read_ulint(
+ const byte* ptr,
+ mlog_id_t type)
+{
+ switch (type) {
+ case MLOG_1BYTE:
+ return(mach_read_from_1(ptr));
+ case MLOG_2BYTES:
+ return(mach_read_from_2(ptr));
+ case MLOG_4BYTES:
+ return(mach_read_from_4(ptr));
+ default:
+ break;
+ }
+
+ ut_error;
+ return(0);
+}
+ibool
+xdes_get_bit(
+/*=========*/
+ const xdes_t* descr, /*!< in: descriptor */
+ ulint bit, /*!< in: XDES_FREE_BIT or XDES_CLEAN_BIT */
+ ulint offset) /*!< in: page offset within extent:
+ 0 ... FSP_EXTENT_SIZE - 1 */
+{
+ ulint index = bit + XDES_BITS_PER_PAGE * offset;
+
+ ulint bit_index = index % 8;
+ ulint byte_index = index / 8;
+
+ return(ut_bit_get_nth(
+ mach_read_ulint(descr + XDES_BITMAP + byte_index,
+ MLOG_1BYTE),
+ bit_index));
+}
+
+/*
+Parse the page and collect/dump the information about page type
+@param [in] page buffer page
+@param [in] xdes xdes page
+@param [in] file file for diagnosis.
+@param [in] page_size page size
+*/
void
parse_page(
-/*=======*/
- uchar* page, /* in: buffer page */
- uchar* xdes) /* in: extend descriptor page */
+ const byte* page,
+ const byte* xdes,
+ FILE* file,
+ page_size_t page_size)
{
- ib_uint64_t id;
- ulint x;
- ulint n_recs;
- ulint page_no;
- ulint left_page_no;
- ulint right_page_no;
- ulint data_bytes;
- int is_leaf;
- int size_range_id;
-
- switch (fil_page_get_type(page)) {
- case FIL_PAGE_INDEX:
- n_fil_page_index++;
- id = btr_page_get_index_id(page);
- n_recs = page_get_n_recs(page);
- page_no = page_get_page_no(page);
- left_page_no = btr_page_get_prev(page);
- right_page_no = btr_page_get_next(page);
- data_bytes = page_get_data_size(page);
- is_leaf = page_is_leaf(page);
- size_range_id = (data_bytes * SIZE_RANGES_FOR_PAGE
- + logical_page_size - 1) /
- logical_page_size;
- if (size_range_id > SIZE_RANGES_FOR_PAGE + 1) {
- /* data_bytes is bigger than logical_page_size */
- size_range_id = SIZE_RANGES_FOR_PAGE + 1;
- }
- if (per_page_details) {
- printf("index " IB_ID_FMT " page " ULINTPF
- " leaf %d n_recs " ULINTPF " data_bytes " ULINTPF
- "\n", id, page_no, is_leaf, n_recs, data_bytes);
- }
- /* update per-index statistics */
- {
- if (index_ids.count(id) == 0) {
- index_ids[id] = per_index_stats();
- }
- std::map<unsigned long long, per_index_stats>::iterator it;
- it = index_ids.find(id);
- per_index_stats &index = (it->second);
- uchar* des = xdes + XDES_ARR_OFFSET
- + XDES_SIZE * ((page_no & (physical_page_size - 1))
- / FSP_EXTENT_SIZE);
- if (xdes_get_bit(des, XDES_FREE_BIT,
- page_no % FSP_EXTENT_SIZE)) {
- index.free_pages++;
- return;
- }
- index.pages++;
- if (is_leaf) {
- index.leaf_pages++;
- if (data_bytes > index.max_data_size) {
- index.max_data_size = data_bytes;
- }
- struct per_page_stats pp(n_recs, data_bytes,
- left_page_no, right_page_no);
-
- index.leaves[page_no] = pp;
-
- if (left_page_no == ULINT32_UNDEFINED) {
- index.first_leaf_page = page_no;
- index.count++;
- }
- }
- index.total_n_recs += n_recs;
- index.total_data_bytes += data_bytes;
- index.pages_in_size_range[size_range_id] ++;
- }
-
- break;
- case FIL_PAGE_UNDO_LOG:
- if (per_page_details) {
- printf("FIL_PAGE_UNDO_LOG\n");
- }
- n_fil_page_undo_log++;
- x = mach_read_from_2(page + TRX_UNDO_PAGE_HDR +
- TRX_UNDO_PAGE_TYPE);
- if (x == TRX_UNDO_INSERT)
- n_undo_insert++;
- else if (x == TRX_UNDO_UPDATE)
- n_undo_update++;
- else
- n_undo_other++;
-
- x = mach_read_from_2(page + TRX_UNDO_SEG_HDR + TRX_UNDO_STATE);
- switch (x) {
- case TRX_UNDO_ACTIVE: n_undo_state_active++; break;
- case TRX_UNDO_CACHED: n_undo_state_cached++; break;
- case TRX_UNDO_TO_FREE: n_undo_state_to_free++; break;
- case TRX_UNDO_TO_PURGE: n_undo_state_to_purge++; break;
- case TRX_UNDO_PREPARED: n_undo_state_prepared++; break;
- default: n_undo_state_other++; break;
- }
- break;
- case FIL_PAGE_INODE:
- if (per_page_details) {
- printf("FIL_PAGE_INODE\n");
- }
- n_fil_page_inode++;
- break;
- case FIL_PAGE_IBUF_FREE_LIST:
- if (per_page_details) {
- printf("FIL_PAGE_IBUF_FREE_LIST\n");
- }
- n_fil_page_ibuf_free_list++;
- break;
- case FIL_PAGE_TYPE_ALLOCATED:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_ALLOCATED\n");
- }
- n_fil_page_type_allocated++;
- break;
- case FIL_PAGE_IBUF_BITMAP:
- if (per_page_details) {
- printf("FIL_PAGE_IBUF_BITMAP\n");
- }
- n_fil_page_ibuf_bitmap++;
- break;
- case FIL_PAGE_TYPE_SYS:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_SYS\n");
- }
- n_fil_page_type_sys++;
- break;
- case FIL_PAGE_TYPE_TRX_SYS:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_TRX_SYS\n");
- }
- n_fil_page_type_trx_sys++;
- break;
- case FIL_PAGE_TYPE_FSP_HDR:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_FSP_HDR\n");
- }
- memcpy(xdes, page, physical_page_size);
- n_fil_page_type_fsp_hdr++;
- break;
- case FIL_PAGE_TYPE_XDES:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_XDES\n");
- }
- memcpy(xdes, page, physical_page_size);
- n_fil_page_type_xdes++;
- break;
- case FIL_PAGE_TYPE_BLOB:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_BLOB\n");
- }
- n_fil_page_type_blob++;
- break;
- case FIL_PAGE_TYPE_ZBLOB:
- case FIL_PAGE_TYPE_ZBLOB2:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_ZBLOB/2\n");
- }
- n_fil_page_type_zblob++;
- break;
+ unsigned long long id=0;
+ ulint undo_page_type=0;
+ ulint n_recs;
+ ulint page_no=0;
+ ulint right_page_no=0;
+ ulint left_page_no=0;
+ ulint data_bytes=0;
+ bool is_leaf=false;
+ ulint size_range_id=0;
+ ulint data_types=0;
+ ulint key_version = 0;
+
+ char str[20]={'\0'};
+
+ /* Check whether page is doublewrite buffer. */
+ if(skip_page) {
+ strcpy(str, "Double_write_buffer");
+ } else {
+ strcpy(str, "-");
+ }
+
+ switch (mach_read_from_2(page + FIL_PAGE_TYPE)) {
+
+ case FIL_PAGE_INDEX:
+ page_type.n_fil_page_index++;
+ id = mach_read_from_8(page + PAGE_HEADER + PAGE_INDEX_ID);
+ n_recs = page_header_get_field(page, PAGE_N_RECS);
+ page_no = page_get_page_no(page);
+ left_page_no = btr_page_get_prev(page);
+ right_page_no = btr_page_get_next(page);
+ key_version = mach_read_from_4(page + FIL_PAGE_FILE_FLUSH_LSN_OR_KEY_VERSION);
+
+ /* If page is encrypted we can't read index header */
+ if (!key_version) {
+ data_bytes = page_get_data_size(page);
+ } else {
+ data_bytes = 0;
+ }
+
+ is_leaf = page_is_leaf(page);
+
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tIndex page\t\t\t|"
+ "\tindex id=%llu,", cur_page_num, id);
+
+ fprintf(file,
+ " page level=" ULINTPF " leaf %u"
+ ", No. of records=" ULINTPF
+ ", garbage=" ULINTPF
+ ", n_recs=" ULINTPF
+ ", %s\n",
+ page_header_get_field(page, PAGE_LEVEL),
+ is_leaf,
+ n_recs,
+ page_header_get_field(page, PAGE_GARBAGE),
+ data_types,
+ str);
+ }
+
+ size_range_id = (data_bytes * SIZE_RANGES_FOR_PAGE
+ + page_size.logical() - 1) /
+ page_size.logical();
+
+ if (size_range_id > SIZE_RANGES_FOR_PAGE + 1) {
+ /* data_bytes is bigger than logical_page_size */
+ size_range_id = SIZE_RANGES_FOR_PAGE + 1;
+ }
+
+ /* update per-index statistics */
+ {
+ if (index_ids.count(id) == 0) {
+ index_ids[id] = per_index_stats();
+ }
+
+ std::map<unsigned long long, per_index_stats>::iterator it;
+ it = index_ids.find(id);
+ per_index_stats &index = (it->second);
+ uchar* des = (uchar *)(xdes + XDES_ARR_OFFSET
+ + XDES_SIZE * ((page_no & (page_size.physical() - 1))
+ / FSP_EXTENT_SIZE));
+
+ if (xdes_get_bit(des, XDES_FREE_BIT,
+ page_no % FSP_EXTENT_SIZE)) {
+ index.free_pages++;
+ return;
+ }
+
+ index.pages++;
+
+ if (is_leaf) {
+ index.leaf_pages++;
+ if (data_bytes > index.max_data_size) {
+ index.max_data_size = data_bytes;
+ }
+
+ struct per_page_stats pp(n_recs, data_bytes,
+ left_page_no, right_page_no);
+
+ index.leaves[page_no] = pp;
+
+ if (left_page_no == ULINT32_UNDEFINED) {
+ index.first_leaf_page = page_no;
+ index.count++;
+ }
+ }
+
+ index.total_n_recs += n_recs;
+ index.total_data_bytes += data_bytes;
+ index.pages_in_size_range[size_range_id] ++;
+ }
+
+ break;
+
+ case FIL_PAGE_UNDO_LOG:
+ page_type.n_fil_page_undo_log++;
+ undo_page_type = mach_read_from_2(page +
+ TRX_UNDO_PAGE_HDR + TRX_UNDO_PAGE_TYPE);
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tUndo log page\t\t\t|",
+ cur_page_num);
+ }
+ if (undo_page_type == TRX_UNDO_INSERT) {
+ page_type.n_undo_insert++;
+ if (page_type_dump) {
+ fprintf(file, "\t%s",
+ "Insert Undo log page");
+ }
+
+ } else if (undo_page_type == TRX_UNDO_UPDATE) {
+ page_type.n_undo_update++;
+ if (page_type_dump) {
+ fprintf(file, "\t%s",
+ "Update undo log page");
+ }
+ }
+
+ undo_page_type = mach_read_from_2(page + TRX_UNDO_SEG_HDR +
+ TRX_UNDO_STATE);
+ switch (undo_page_type) {
+ case TRX_UNDO_ACTIVE:
+ page_type.n_undo_state_active++;
+ if (page_type_dump) {
+ fprintf(file, ", %s", "Undo log of "
+ "an active transaction");
+ }
+ break;
+
+ case TRX_UNDO_CACHED:
+ page_type.n_undo_state_cached++;
+ if (page_type_dump) {
+ fprintf(file, ", %s", "Page is "
+ "cached for quick reuse");
+ }
+ break;
+
+ case TRX_UNDO_TO_FREE:
+ page_type.n_undo_state_to_free++;
+ if (page_type_dump) {
+ fprintf(file, ", %s", "Insert undo "
+ "segment that can be freed");
+ }
+ break;
+
+ case TRX_UNDO_TO_PURGE:
+ page_type.n_undo_state_to_purge++;
+ if (page_type_dump) {
+ fprintf(file, ", %s", "Will be "
+ "freed in purge when all undo"
+ "data in it is removed");
+ }
+ break;
+
+ case TRX_UNDO_PREPARED:
+ page_type.n_undo_state_prepared++;
+ if (page_type_dump) {
+ fprintf(file, ", %s", "Undo log of "
+ "an prepared transaction");
+ }
+ break;
+
+ default:
+ page_type.n_undo_state_other++;
+ break;
+ }
+ if(page_type_dump) {
+ fprintf(file, ", %s\n", str);
+ }
+ break;
+
+ case FIL_PAGE_INODE:
+ page_type.n_fil_page_inode++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tInode page\t\t\t|"
+ "\t%s\n",cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_IBUF_FREE_LIST:
+ page_type.n_fil_page_ibuf_free_list++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tInsert buffer free list"
+ " page\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_ALLOCATED:
+ page_type.n_fil_page_type_allocated++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tFreshly allocated "
+ "page\t\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_IBUF_BITMAP:
+ page_type.n_fil_page_ibuf_bitmap++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tInsert Buffer "
+ "Bitmap\t\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_SYS:
+ page_type.n_fil_page_type_sys++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tSystem page\t\t\t|"
+ "\t%s\n",cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_TRX_SYS:
+ page_type.n_fil_page_type_trx_sys++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tTransaction system "
+ "page\t\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_FSP_HDR:
+ page_type.n_fil_page_type_fsp_hdr++;
+ memcpy((void *)xdes, (void *)page, page_size.physical());
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tFile Space "
+ "Header\t\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_XDES:
+ page_type.n_fil_page_type_xdes++;
+ memcpy((void *)xdes, (void *)page, page_size.physical());
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tExtent descriptor "
+ "page\t\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_BLOB:
+ page_type.n_fil_page_type_blob++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tBLOB page\t\t\t|\t%s\n",
+ cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_ZBLOB:
+ page_type.n_fil_page_type_zblob++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tCompressed BLOB "
+ "page\t\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
+ case FIL_PAGE_TYPE_ZBLOB2:
+ page_type.n_fil_page_type_zblob2++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tSubsequent Compressed "
+ "BLOB page\t|\t%s\n", cur_page_num, str);
+ }
+ break;
+
case FIL_PAGE_PAGE_COMPRESSED:
- if (per_page_details) {
- printf("FIL_PAGE_PAGE_COMPRESSED\n");
+ page_type.n_fil_page_type_page_compressed++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tPage compressed "
+ "page\t|\t%s\n", cur_page_num, str);
}
- n_fil_page_type_page_compressed++;
break;
case FIL_PAGE_PAGE_COMPRESSED_ENCRYPTED:
- if (per_page_details) {
- printf("FIL_PAGE_PAGE_COMPRESSED_ENCRYPTED\n");
+ page_type.n_fil_page_type_page_compressed_encrypted++;
+ if (page_type_dump) {
+ fprintf(file, "#::%8" PRIuMAX "\t\t|\t\tPage compressed encrypted "
+ "page\t|\t%s\n", cur_page_num, str);
}
- n_fil_page_type_page_compressed_encrypted++;
break;
- default:
- if (per_page_details) {
- printf("FIL_PAGE_TYPE_OTHER\n");
- }
- n_fil_page_type_other++;
- }
+ default:
+ page_type.n_fil_page_type_other++;
+ break;
+ }
}
+/**
+@param [in/out] file_name name of the filename
-void print_index_leaf_stats(unsigned long long id, const per_index_stats& index)
+@retval FILE pointer if successfully created else NULL when error occured.
+*/
+FILE*
+create_file(
+ char* file_name)
{
- ulint page_no = index.first_leaf_page;
- std::map<ulint, per_page_stats>::const_iterator it_page = index.leaves.find(page_no);
- printf("\nindex: %llu leaf page stats: n_pages = %llu\n",
- id, index.leaf_pages);
- printf("page_no\tdata_size\tn_recs\n");
- while (it_page != index.leaves.end()) {
- const per_page_stats& stat = it_page->second;
- printf("%lu\t%lu\t%lu\n", it_page->first, stat.data_size, stat.n_recs);
- page_no = stat.right_page_no;
- it_page = index.leaves.find(page_no);
- }
+ FILE* file = NULL;
+
+#ifndef _WIN32
+ file = fopen(file_name, "wb");
+ if (file == NULL) {
+ fprintf(stderr, "Failed to create file: %s: %s\n",
+ file_name, strerror(errno));
+ return(NULL);
+ }
+#else
+ HANDLE hFile; /* handle to open file. */
+ int fd = 0;
+ hFile = CreateFile((LPCTSTR) file_name,
+ GENERIC_READ | GENERIC_WRITE,
+ FILE_SHARE_READ | FILE_SHARE_DELETE,
+ NULL, CREATE_NEW, NULL, NULL);
+
+ if (hFile == INVALID_HANDLE_VALUE) {
+ /* print the error message. */
+ fprintf(stderr, "Filename::%s %s\n",
+ file_name,
+ error_message(GetLastError()));
+
+ return(NULL);
+ }
+
+ /* get the file descriptor. */
+ fd= _open_osfhandle((intptr_t)hFile, _O_RDWR | _O_BINARY);
+ file = fdopen(fd, "wb");
+#endif /* _WIN32 */
+
+ return(file);
}
-void defrag_analysis(unsigned long long id, const per_index_stats& index)
+/*
+ Print the page type count of a tablespace.
+ @param [in] fil_out stream where the output goes.
+*/
+void
+print_summary(
+ FILE* fil_out)
{
- // TODO: make it work for compressed pages too
- std::map<ulint, per_page_stats>::const_iterator it = index.leaves.find(index.first_leaf_page);
- ulint n_pages = 0;
- ulint n_leaf_pages = 0;
- while (it != index.leaves.end()) {
- ulint data_size_total = 0;
- for (ulong i = 0; i < n_merge; i++) {
- const per_page_stats& stat = it->second;
- n_leaf_pages ++;
- data_size_total += stat.data_size;
- it = index.leaves.find(stat.right_page_no);
- if (it == index.leaves.end()) {
- break;
- }
- }
- if (index.max_data_size) {
- n_pages += data_size_total / index.max_data_size;
- if (data_size_total % index.max_data_size != 0) {
- n_pages += 1;
- }
- }
- }
- if (index.leaf_pages)
- printf("count = %lu free = %lu\n", index.count, index.free_pages);
- printf("%llu\t\t%llu\t\t%lu\t\t%lu\t\t%lu\t\t%.2f\t%lu\n",
- id, index.leaf_pages, n_leaf_pages, n_merge, n_pages,
- 1.0 - (double)n_pages / (double)n_leaf_pages, index.max_data_size);
+ fprintf(fil_out, "\n================PAGE TYPE SUMMARY==============\n");
+ fprintf(fil_out, "#PAGE_COUNT\tPAGE_TYPE");
+ fprintf(fil_out, "\n===============================================\n");
+ fprintf(fil_out, "%8d\tIndex page\n",
+ page_type.n_fil_page_index);
+ fprintf(fil_out, "%8d\tUndo log page\n",
+ page_type.n_fil_page_undo_log);
+ fprintf(fil_out, "%8d\tInode page\n",
+ page_type.n_fil_page_inode);
+ fprintf(fil_out, "%8d\tInsert buffer free list page\n",
+ page_type.n_fil_page_ibuf_free_list);
+ fprintf(fil_out, "%8d\tFreshly allocated page\n",
+ page_type.n_fil_page_type_allocated);
+ fprintf(fil_out, "%8d\tInsert buffer bitmap\n",
+ page_type.n_fil_page_ibuf_bitmap);
+ fprintf(fil_out, "%8d\tSystem page\n",
+ page_type.n_fil_page_type_sys);
+ fprintf(fil_out, "%8d\tTransaction system page\n",
+ page_type.n_fil_page_type_trx_sys);
+ fprintf(fil_out, "%8d\tFile Space Header\n",
+ page_type.n_fil_page_type_fsp_hdr);
+ fprintf(fil_out, "%8d\tExtent descriptor page\n",
+ page_type.n_fil_page_type_xdes);
+ fprintf(fil_out, "%8d\tBLOB page\n",
+ page_type.n_fil_page_type_blob);
+ fprintf(fil_out, "%8d\tCompressed BLOB page\n",
+ page_type.n_fil_page_type_zblob);
+ fprintf(fil_out, "%8d\tPage compressed page\n",
+ page_type.n_fil_page_type_page_compressed);
+ fprintf(fil_out, "%8d\tPage compressed encrypted page\n",
+ page_type.n_fil_page_type_page_compressed_encrypted);
+ fprintf(fil_out, "%8d\tOther type of page",
+ page_type.n_fil_page_type_other);
+
+ fprintf(fil_out, "\n===============================================\n");
+ fprintf(fil_out, "Additional information:\n");
+ fprintf(fil_out, "Undo page type: %d insert, %d update, %d other\n",
+ page_type.n_undo_insert,
+ page_type.n_undo_update,
+ page_type.n_undo_other);
+ fprintf(fil_out, "Undo page state: %d active, %d cached, %d to_free, %d"
+ " to_purge, %d prepared, %d other\n",
+ page_type.n_undo_state_active,
+ page_type.n_undo_state_cached,
+ page_type.n_undo_state_to_free,
+ page_type.n_undo_state_to_purge,
+ page_type.n_undo_state_prepared,
+ page_type.n_undo_state_other);
+
+ fprintf(fil_out, "index_id\t#pages\t\t#leaf_pages\t#recs_per_page"
+ "\t#bytes_per_page\n");
+
+ for (std::map<unsigned long long, per_index_stats>::const_iterator it = index_ids.begin();
+ it != index_ids.end(); it++) {
+ const per_index_stats& index = it->second;
+ fprintf(fil_out, "%lld\t\t%lld\t\t%lld\t\t%lld\t\t%lld\n",
+ it->first, index.pages, index.leaf_pages,
+ index.total_n_recs / index.pages,
+ index.total_data_bytes / index.pages);
+ }
+
+ fprintf(fil_out, "\n");
+ fprintf(fil_out, "index_id\tpage_data_bytes_histgram(empty,...,oversized)\n");
+
+ for (std::map<unsigned long long, per_index_stats>::const_iterator it = index_ids.begin();
+ it != index_ids.end(); it++) {
+ fprintf(fil_out, "%lld\t", it->first);
+ const per_index_stats& index = it->second;
+ for (ulint i = 0; i < SIZE_RANGES_FOR_PAGE+2; i++) {
+ fprintf(fil_out, "\t%lld", index.pages_in_size_range[i]);
+ }
+ fprintf(fil_out, "\n");
+ }
+
+ if (do_leaf) {
+ print_leaf_stats(fil_out);
+ }
}
-void print_leaf_stats()
+/* command line argument for innochecksum tool. */
+static struct my_option innochecksum_options[] = {
+ {"help", '?', "Displays this help and exits.",
+ 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"info", 'I', "Synonym for --help.",
+ 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"version", 'V', "Displays version information and exits.",
+ 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"verbose", 'v', "Verbose (prints progress every 5 seconds).",
+ &verbose, &verbose, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"debug", '#', "Output debug log. See " REFMAN "dbug-package.html",
+ &dbug_setting, &dbug_setting, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0},
+ {"count", 'c', "Print the count of pages in the file and exits.",
+ &just_count, &just_count, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"start_page", 's', "Start on this page number (0 based).",
+ &start_page, &start_page, 0, GET_ULL, REQUIRED_ARG,
+ 0, 0, ULLONG_MAX, 0, 1, 0},
+ {"end_page", 'e', "End at this page number (0 based).",
+ &end_page, &end_page, 0, GET_ULL, REQUIRED_ARG,
+ 0, 0, ULLONG_MAX, 0, 1, 0},
+ {"page", 'p', "Check only this page (0 based).",
+ &do_page, &do_page, 0, GET_ULL, REQUIRED_ARG,
+ 0, 0, ULLONG_MAX, 0, 1, 0},
+ {"strict-check", 'C', "Specify the strict checksum algorithm by the user.",
+ &strict_check, &strict_check, &innochecksum_algorithms_typelib,
+ GET_ENUM, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
+ {"no-check", 'n', "Ignore the checksum verification.",
+ &no_check, &no_check, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"allow-mismatches", 'a', "Maximum checksum mismatch allowed.",
+ &allow_mismatches, &allow_mismatches, 0,
+ GET_ULL, REQUIRED_ARG, 0, 0, ULLONG_MAX, 0, 1, 0},
+ {"write", 'w', "Rewrite the checksum algorithm by the user.",
+ &write_check, &write_check, &innochecksum_algorithms_typelib,
+ GET_ENUM, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
+ {"page-type-summary", 'S', "Display a count of each page type "
+ "in a tablespace.", &page_type_summary, &page_type_summary, 0,
+ GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"page-type-dump", 'D', "Dump the page type info for each page in a "
+ "tablespace.", &page_dump_filename, &page_dump_filename, 0,
+ GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
+ {"log", 'l', "log output.",
+ &log_filename, &log_filename, 0,
+ GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
+ {"leaf", 'e', "Examine leaf index pages",
+ &do_leaf, &do_leaf, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
+ {"merge", 'm', "leaf page count if merge given number of consecutive pages",
+ &n_merge, &n_merge, 0, GET_ULONG, REQUIRED_ARG, 0, 0, (longlong)10L, 0, 1, 0},
+
+ {0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}
+};
+
+/* Print out the Innodb version and machine information. */
+static void print_version(void)
{
- printf("\n**************************************************\n");
- printf("index_id\t#leaf_pages\t#actual_leaf_pages\tn_merge\t"
- "#leaf_after_merge\tdefrag\n");
- for (std::map<unsigned long long, per_index_stats>::const_iterator it = index_ids.begin(); it != index_ids.end(); it++) {
- const per_index_stats& index = it->second;
- if (verbose) {
- print_index_leaf_stats(it->first, index);
- }
- if (n_merge) {
- defrag_analysis(it->first, index);
- }
- }
+#ifdef DBUG_OFF
+ printf("%s Ver %s, for %s (%s)\n",
+ my_progname, INNODB_VERSION_STR,
+ SYSTEM_TYPE, MACHINE_TYPE);
+#else
+ printf("%s-debug Ver %s, for %s (%s)\n",
+ my_progname, INNODB_VERSION_STR,
+ SYSTEM_TYPE, MACHINE_TYPE);
+#endif /* DBUG_OFF */
}
-void
-print_stats()
-/*========*/
+static void usage(void)
{
- unsigned long long i;
-
- printf("%d\tbad checksum\n", n_bad_checksum);
- printf("%d\tFIL_PAGE_INDEX\n", n_fil_page_index);
- printf("%d\tFIL_PAGE_UNDO_LOG\n", n_fil_page_undo_log);
- printf("%d\tFIL_PAGE_INODE\n", n_fil_page_inode);
- printf("%d\tFIL_PAGE_IBUF_FREE_LIST\n", n_fil_page_ibuf_free_list);
- printf("%d\tFIL_PAGE_TYPE_ALLOCATED\n", n_fil_page_type_allocated);
- printf("%d\tFIL_PAGE_IBUF_BITMAP\n", n_fil_page_ibuf_bitmap);
- printf("%d\tFIL_PAGE_TYPE_SYS\n", n_fil_page_type_sys);
- printf("%d\tFIL_PAGE_TYPE_TRX_SYS\n", n_fil_page_type_trx_sys);
- printf("%d\tFIL_PAGE_TYPE_FSP_HDR\n", n_fil_page_type_fsp_hdr);
- printf("%d\tFIL_PAGE_TYPE_XDES\n", n_fil_page_type_xdes);
- printf("%d\tFIL_PAGE_TYPE_BLOB\n", n_fil_page_type_blob);
- printf("%d\tFIL_PAGE_TYPE_ZBLOB\n", n_fil_page_type_zblob);
- printf("%d\tFIL_PAGE_PAGE_COMPRESSED\n", n_fil_page_type_page_compressed);
- printf("%d\tFIL_PAGE_PAGE_COMPRESSED_ENCRYPTED\n", n_fil_page_type_page_compressed_encrypted);
- printf("%d\tother\n", n_fil_page_type_other);
- printf("%d\tmax index_id\n", n_fil_page_max_index_id);
- printf("undo type: %d insert, %d update, %d other\n",
- n_undo_insert, n_undo_update, n_undo_other);
- printf("undo state: %d active, %d cached, %d to_free, %d to_purge,"
- " %d prepared, %d other\n", n_undo_state_active,
- n_undo_state_cached, n_undo_state_to_free,
- n_undo_state_to_purge, n_undo_state_prepared,
- n_undo_state_other);
-
- printf("index_id\t#pages\t\t#leaf_pages\t#recs_per_page"
- "\t#bytes_per_page\n");
- for (std::map<unsigned long long, per_index_stats>::const_iterator it = index_ids.begin(); it != index_ids.end(); it++) {
- const per_index_stats& index = it->second;
- longlong recs_per_page = index.total_n_recs;
- longlong bytes_per_page = index.total_data_bytes;
- if (index.total_n_recs && index.pages) {
- recs_per_page = index.total_n_recs / index.pages;
- }
- if (index.total_data_bytes && index.pages) {
- bytes_per_page = index.total_data_bytes / index.pages;
- }
- printf("%lld\t\t%lld\t\t%lld\t\t%lld\t\t%lld\n",
- it->first, index.pages, index.leaf_pages,
- recs_per_page,
- bytes_per_page);
- }
- printf("\n");
- printf("index_id\tpage_data_bytes_histgram(empty,...,oversized)\n");
- for (std::map<unsigned long long, per_index_stats>::const_iterator it = index_ids.begin(); it != index_ids.end(); it++) {
- printf("%lld\t", it->first);
- const per_index_stats& index = it->second;
- for (i = 0; i < SIZE_RANGES_FOR_PAGE+2; i++) {
- printf("\t%lld", index.pages_in_size_range[i]);
- }
- printf("\n");
- }
- if (do_leaf) {
- print_leaf_stats();
- }
+ print_version();
+ puts(ORACLE_WELCOME_COPYRIGHT_NOTICE("2000"));
+ printf("InnoDB offline file checksum utility.\n");
+ printf("Usage: %s [-c] [-s <start page>] [-e <end page>] "
+ "[-p <page>] [-v] [-a <allow mismatches>] [-n] "
+ "[-C <strict-check>] [-w <write>] [-S] [-D <page type dump>] "
+ "[-l <log>] [-e] <filename or [-]>\n", my_progname);
+ printf("See " REFMAN "innochecksum.html for usage hints.\n");
+ my_print_help(innochecksum_options);
+ my_print_variables(innochecksum_options);
}
-int main(int argc, char **argv)
+extern "C" my_bool
+innochecksum_get_one_option(
+ int optid,
+ const struct my_option *opt MY_ATTRIBUTE((unused)),
+ char *argument MY_ATTRIBUTE((unused)))
{
- FILE* f; /* our input file */
- char* filename; /* our input filename. */
- unsigned char *big_buf= 0, *buf;
- unsigned char *big_xdes= 0, *xdes;
- ulong bytes; /* bytes read count */
- ulint ct; /* current page number (0 based) */
- time_t now; /* current time */
- time_t lastt; /* last time */
- ulint oldcsum, oldcsumfield, csum, csumfield, crc32, logseq, logseqfield;
- /* ulints for checksum storage */
- unsigned long long int size; /* size of file (has to be 64 bits) */
- ulint pages; /* number of pages in file */
- long long offset= 0;
- int fd;
-
- printf("InnoDB offline file checksum utility.\n");
-
- ut_crc32_init();
-
- MY_INIT(argv[0]);
-
- if (get_options(&argc,&argv))
- exit(1);
-
- if (verbose)
- my_print_variables(innochecksum_options);
-
- /* The file name is not optional */
- filename = *argv;
- if (*filename == '\0')
- {
- fprintf(stderr, "Error; File name missing\n");
- goto error_out;
- }
-
-#ifdef _WIN32
- /* Switch off OS file buffering for the file. */
-
- HANDLE h = CreateFile(filename, GENERIC_READ,
- FILE_SHARE_READ|FILE_SHARE_WRITE, 0,
- OPEN_EXISTING, FILE_FLAG_NO_BUFFERING, 0);
-
- if (!h)
- {
- fprintf(stderr, "Error; cant open file\n");
- goto error;
- }
-
- if (!GetFileSizeEx(h, (LARGE_INTEGER *)&size))
- {
- fprintf(stderr, "Error; GetFileSize() failed\n");
- goto error;
- }
+ switch (optid) {
+#ifndef DBUG_OFF
+ case '#':
+ dbug_setting = argument
+ ? argument
+ : IF_WIN("d:O,innochecksum.trace",
+ "d:o,/tmp/innochecksum.trace");
+ DBUG_PUSH(dbug_setting);
+ break;
+#endif /* !DBUG_OFF */
+ case 'e':
+ use_end_page = true;
+ break;
+ case 'p':
+ end_page = start_page = do_page;
+ use_end_page = true;
+ do_one_page = true;
+ break;
+ case 'V':
+ print_version();
+ my_end(0);
+ exit(EXIT_SUCCESS);
+ break;
+ case 'C':
+ strict_verify = true;
+ switch ((srv_checksum_algorithm_t) strict_check) {
+
+ case SRV_CHECKSUM_ALGORITHM_STRICT_CRC32:
+ case SRV_CHECKSUM_ALGORITHM_CRC32:
+ srv_checksum_algorithm =
+ SRV_CHECKSUM_ALGORITHM_STRICT_CRC32;
+ break;
+
+ case SRV_CHECKSUM_ALGORITHM_STRICT_INNODB:
+ case SRV_CHECKSUM_ALGORITHM_INNODB:
+ srv_checksum_algorithm =
+ SRV_CHECKSUM_ALGORITHM_STRICT_INNODB;
+ break;
+
+ case SRV_CHECKSUM_ALGORITHM_STRICT_NONE:
+ case SRV_CHECKSUM_ALGORITHM_NONE:
+ srv_checksum_algorithm =
+ SRV_CHECKSUM_ALGORITHM_STRICT_NONE;
+ break;
+ default:
+ return(true);
+ }
+ break;
+ case 'n':
+ no_check = true;
+ break;
+ case 'a':
+ case 'S':
+ break;
+ case 'w':
+ do_write = true;
+ break;
+ case 'D':
+ page_type_dump = true;
+ break;
+ case 'l':
+ is_log_enabled = true;
+ break;
+ case 'I':
+ case '?':
+ usage();
+ my_end(0);
+ exit(EXIT_SUCCESS);
+ break;
+ }
- fd = _open_osfhandle ((intptr_t) h, _O_RDONLY);
- if (fd < 0)
- {
- fprintf(stderr, "Error; _open_osfhandle() failed\n");
- goto error;
- }
+ return(false);
+}
- f = _fdopen(fd, "rb");
- if (!f)
- {
- fprintf(stderr, "Error; fdopen() failed\n");
- goto error;
- }
+static
+bool
+get_options(
+ int *argc,
+ char ***argv)
+{
+ if (handle_options(argc, argv, innochecksum_options,
+ innochecksum_get_one_option)) {
+ my_end(0);
+ exit(true);
+ }
- /*
- Disable stdio buffering (FILE_FLAG_NO_BUFFERING requires properly IO buffers
- which stdio does not guarantee.
- */
- setvbuf(f, NULL, _IONBF, 0);
+ /* The next arg must be the filename */
+ if (!*argc) {
+ usage();
+ my_end(0);
+ return (true);
+ }
-#else
- struct stat st;
- /* stat the file to get size and page count */
- if (stat(filename, &st))
- {
- fprintf(stderr, "Error; %s cannot be found\n", filename);
- goto error_out;
- }
- size= st.st_size;
+ return (false);
+}
- /* Open the file for reading */
- f= fopen(filename, "rb");
+int main(
+ int argc,
+ char **argv)
+{
+ /* our input file. */
+ FILE* fil_in = NULL;
+ /* our input filename. */
+ char* filename;
+ /* Buffer to store pages read. */
+ byte* buf = NULL;
+ /* Buffer for xdes */
+ byte* xdes = NULL;
+ /* bytes read count */
+ ulong bytes;
+ /* Buffer to decompress page.*/
+#ifdef MYSQL_COMPRESSION
+ byte* tbuf = NULL;
#endif
+ /* current time */
+ time_t now;
+ /* last time */
+ time_t lastt;
+ /* stat, to get file size. */
+#ifdef _WIN32
+ struct _stat64 st;
+#else
+ struct stat st;
+#endif /* _WIN32 */
+
+ /* size of file (has to be 64 bits) */
+ unsigned long long int size = 0;
+ /* number of pages in file */
+ ulint pages;
+
+ off_t offset = 0;
+ /* count the no. of page corrupted. */
+ ulint mismatch_count = 0;
+ /* Variable to ack the page is corrupted or not. */
+ bool is_corrupted = false;
+
+ bool partial_page_read = false;
+ /* Enabled when read from stdin is done. */
+ bool read_from_stdin = false;
+ FILE* fil_page_type = NULL;
+ fpos_t pos;
+
+ /* Use to check the space id of given file. If space_id is zero,
+ then check whether page is doublewrite buffer.*/
+ ulint space_id = 0UL;
+ /* enable when space_id of given file is zero. */
+ bool is_system_tablespace = false;
+
+ ut_crc32_init();
+ MY_INIT(argv[0]);
+ DBUG_ENTER("main");
+ DBUG_PROCESS(argv[0]);
+
+ if (get_options(&argc,&argv)) {
+ DBUG_RETURN(1);
+ }
- if (f == NULL)
- {
- fprintf(stderr, "Error; %s cannot be opened", filename);
- perror(" ");
- goto error_out;
- }
+ if (strict_verify && no_check) {
+ fprintf(stderr, "Error: --strict-check option cannot be used "
+ "together with --no-check option.\n");
+ DBUG_RETURN(1);
+ }
- big_buf = (unsigned char *)malloc(2 * UNIV_PAGE_SIZE_MAX);
- if (big_buf == NULL)
- {
- fprintf(stderr, "Error; failed to allocate memory\n");
- perror("");
- goto error_f;
- }
+ if (no_check && !do_write) {
+ fprintf(stderr, "Error: --no-check must be associated with "
+ "--write option.\n");
+ DBUG_RETURN(1);
+ }
- /* Make sure the page is aligned */
- buf = (unsigned char*)ut_align_down(big_buf
- + UNIV_PAGE_SIZE_MAX, UNIV_PAGE_SIZE_MAX);
+ if (page_type_dump) {
+ fil_page_type = create_file(page_dump_filename);
+ if (!fil_page_type) {
+ DBUG_RETURN(1);
+ }
+ }
- big_xdes = (unsigned char *)malloc(2 * UNIV_PAGE_SIZE_MAX);
- if (big_xdes == NULL)
- {
- fprintf(stderr, "Error; failed to allocate memory\n");
- perror("");
- goto error_big_buf;
- }
+ if (is_log_enabled) {
+ log_file = create_file(log_filename);
+ if (!log_file) {
+ DBUG_RETURN(1);
+ }
+ fprintf(log_file, "InnoDB File Checksum Utility.\n");
+ }
- /* Make sure the page is aligned */
- xdes = (unsigned char*)ut_align_down(big_xdes
- + UNIV_PAGE_SIZE_MAX, UNIV_PAGE_SIZE_MAX);
+ if (verbose) {
+ my_print_variables(innochecksum_options);
+ }
- if (!get_page_size(f, buf, &logical_page_size, &physical_page_size))
- goto error;
+ buf = (byte*) malloc(UNIV_PAGE_SIZE_MAX * 2);
+ xdes = (byte *) malloc(UNIV_PAGE_SIZE_MAX *2);
+#ifdef MYSQL_COMPRESSION
+ tbuf = buf + UNIV_PAGE_SIZE_MAX;
+#endif
+ /* The file name is not optional. */
+ for (int i = 0; i < argc; ++i) {
+ /* Reset parameters for each file. */
+ filename = argv[i];
+ memset(&page_type, 0, sizeof(innodb_page_type));
+ is_corrupted = false;
+ partial_page_read = false;
+ skip_page = false;
+
+ if (is_log_enabled) {
+ fprintf(log_file, "Filename = %s\n", filename);
+ }
+
+ if (*filename == '-') {
+ /* read from stdin. */
+ fil_in = stdin;
+ read_from_stdin = true;
+
+ }
+
+ /* stat the file to get size and page count. */
+ if (!read_from_stdin &&
+#ifdef _WIN32
+ _stat64(filename, &st)) {
+#else
+ stat(filename, &st)) {
+#endif /* _WIN32 */
+ fprintf(stderr, "Error: %s cannot be found\n",
+ filename);
+
+ goto err_exit;
+ }
+
+ if (!read_from_stdin) {
+ size = st.st_size;
+ fil_in = open_file(filename);
+ /*If fil_in is NULL, terminate as some error encountered */
+ if(fil_in == NULL) {
+ goto err_exit;
+ }
+ /* Save the current file pointer in pos variable.*/
+ if (0 != fgetpos(fil_in, &pos)) {
+ perror("fgetpos");
+ goto err_exit;
+ }
+ }
+
+ /* Testing for lock mechanism. The innochecksum
+ acquire lock on given file. So other tools accessing the same
+ file for processsing must fail. */
+#ifdef _WIN32
+ DBUG_EXECUTE_IF("innochecksum_cause_mysqld_crash",
+ ut_ad(page_dump_filename);
+ while((_access( page_dump_filename, 0)) == 0) {
+ sleep(1);
+ }
+ DBUG_RETURN(0); );
+#else
+ DBUG_EXECUTE_IF("innochecksum_cause_mysqld_crash",
+ ut_ad(page_dump_filename);
+ struct stat status_buf;
+ while(stat(page_dump_filename, &status_buf) == 0) {
+ sleep(1);
+ }
+ DBUG_RETURN(0); );
+#endif /* _WIN32 */
+
+ /* Read the minimum page size. */
+ bytes = ulong(fread(buf, 1, UNIV_ZIP_SIZE_MIN, fil_in));
+ partial_page_read = true;
+
+ if (bytes != UNIV_ZIP_SIZE_MIN) {
+ fprintf(stderr, "Error: Was not able to read the "
+ "minimum page size ");
+ fprintf(stderr, "of %d bytes. Bytes read was %lu\n",
+ UNIV_ZIP_SIZE_MIN, bytes);
+
+ goto err_exit;
+ }
+
+ /* enable variable is_system_tablespace when space_id of given
+ file is zero. Use to skip the checksum verification and rewrite
+ for doublewrite pages. */
+ is_system_tablespace = (!memcmp(&space_id, buf +
+ FIL_PAGE_ARCH_LOG_NO_OR_SPACE_ID, 4))
+ ? true : false;
+
+ const page_size_t& page_size = get_page_size(buf);
+
+ pages = (ulint) (size / page_size.physical());
+
+ if (just_count) {
+ if (read_from_stdin) {
+ fprintf(stderr, "Number of pages:" ULINTPF "\n", pages);
+ } else {
+ printf("Number of pages:" ULINTPF "\n", pages);
+ }
+ continue;
+ } else if (verbose && !read_from_stdin) {
+ if (is_log_enabled) {
+ fprintf(log_file, "file %s = %llu bytes "
+ "(" ULINTPF " pages)\n", filename, size, pages);
+ if (do_one_page) {
+ fprintf(log_file, "Innochecksum: "
+ "checking page %" PRIuMAX "\n",
+ do_page);
+ }
+ }
+ } else {
+ if (is_log_enabled) {
+ fprintf(log_file, "Innochecksum: checking "
+ "pages in range %" PRIuMAX " to %" PRIuMAX "\n",
+ start_page, use_end_page ?
+ end_page : (pages - 1));
+ }
+ }
+
+ /* seek to the necessary position */
+ if (start_page) {
+ if (!read_from_stdin) {
+ /* If read is not from stdin, we can use
+ fseeko() to position the file pointer to
+ the desired page. */
+ partial_page_read = false;
+
+ offset = (off_t) start_page
+ * (off_t) page_size.physical();
+#ifdef _WIN32
+ if (_fseeki64(fil_in, offset, SEEK_SET)) {
+#else
+ if (fseeko(fil_in, offset, SEEK_SET)) {
+#endif /* _WIN32 */
+ perror("Error: Unable to seek to "
+ "necessary offset");
+
+ goto err_exit;
+ }
+ /* Save the current file pointer in
+ pos variable. */
+ if (0 != fgetpos(fil_in, &pos)) {
+ perror("fgetpos");
+
+ goto err_exit;
+ }
+ } else {
+
+ ulong count = 0;
+
+ while (!feof(fil_in)) {
+ if (start_page == count) {
+ break;
+ }
+ /* We read a part of page to find the
+ minimum page size. We cannot reset
+ the file pointer to the beginning of
+ the page if we are reading from stdin
+ (fseeko() on stdin doesn't work). So
+ read only the remaining part of page,
+ if partial_page_read is enable. */
+ bytes = read_file(buf,
+ partial_page_read,
+ static_cast<ulong>(
+ page_size.physical()),
+ fil_in);
+
+ partial_page_read = false;
+ count++;
+
+ if (!bytes || feof(fil_in)) {
+ fprintf(stderr, "Error: Unable "
+ "to seek to necessary "
+ "offset");
+
+ goto err_exit;
+ }
+ }
+ }
+ }
+
+ if (page_type_dump) {
+ fprintf(fil_page_type,
+ "\n\nFilename::%s\n", filename);
+ fprintf(fil_page_type,
+ "========================================"
+ "======================================\n");
+ fprintf(fil_page_type,
+ "\tPAGE_NO\t\t|\t\tPAGE_TYPE\t\t"
+ "\t|\tEXTRA INFO\n");
+ fprintf(fil_page_type,
+ "========================================"
+ "======================================\n");
+ }
+
+ /* main checksumming loop */
+ cur_page_num = start_page;
+ lastt = 0;
+ while (!feof(fil_in)) {
+
+ bytes = read_file(buf, partial_page_read,
+ static_cast<ulong>(
+ page_size.physical()), fil_in);
+ partial_page_read = false;
+
+ if (!bytes && feof(fil_in)) {
+ break;
+ }
+
+ if (ferror(fil_in)) {
+ fprintf(stderr, "Error reading " ULINTPF " bytes",
+ page_size.physical());
+ perror(" ");
+
+ goto err_exit;
+ }
+
+ if (bytes != page_size.physical()) {
+ fprintf(stderr, "Error: bytes read (%lu) "
+ "doesn't match page size (" ULINTPF ")\n",
+ bytes, page_size.physical());
+ goto err_exit;
+ }
+
+ if (is_system_tablespace) {
+ /* enable when page is double write buffer.*/
+ skip_page = is_page_doublewritebuffer(buf);
+ } else {
+ skip_page = false;
+#ifdef MYSQL_COMPRESSION
+ if (!page_decompress(buf, tbuf, page_size)) {
+
+ fprintf(stderr,
+ "Page decompress failed");
+
+ goto err_exit;
+ }
+#endif
+ }
+
+ ulint page_type = mach_read_from_2(buf + FIL_PAGE_TYPE);
+
+ if (page_type == FIL_PAGE_PAGE_COMPRESSED ||
+ page_type == FIL_PAGE_PAGE_COMPRESSED_ENCRYPTED) {
+ skip_page = true;
+ }
+
+ /* If no-check is enabled, skip the
+ checksum verification.*/
+ if (!no_check) {
+ /* Checksum verification */
+ if (!skip_page) {
+ is_corrupted = is_page_corrupted(
+ buf, page_size);
+
+ if (is_corrupted) {
+ fprintf(stderr, "Fail: page "
+ "%" PRIuMAX " invalid\n",
+ cur_page_num);
+
+ mismatch_count++;
+
+ if(mismatch_count > allow_mismatches) {
+ fprintf(stderr,
+ "Exceeded the "
+ "maximum allowed "
+ "checksum mismatch "
+ "count::%" PRIuMAX "\n",
+ allow_mismatches);
+
+ goto err_exit;
+ }
+ }
+ }
+ }
+
+ /* Rewrite checksum */
+ if (do_write
+ && !write_file(filename, fil_in, buf,
+ page_size.is_compressed(), &pos,
+ static_cast<ulong>(page_size.physical()))) {
+
+ goto err_exit;
+ }
+
+ /* end if this was the last page we were supposed to check */
+ if (use_end_page && (cur_page_num >= end_page)) {
+ break;
+ }
+
+ if (page_type_summary || page_type_dump) {
+ parse_page(buf, xdes , fil_page_type, page_size);
+ }
+
+ /* do counter increase and progress printing */
+ cur_page_num++;
+ if (verbose && !read_from_stdin) {
+ if ((cur_page_num % 64) == 0) {
+ now = time(0);
+ if (!lastt) {
+ lastt= now;
+ }
+ if (now - lastt >= 1
+ && is_log_enabled) {
+ fprintf(log_file, "page %" PRIuMAX " "
+ "okay: %.3f%% done\n",
+ (cur_page_num - 1),
+ (float) cur_page_num / pages * 100);
+ lastt = now;
+ }
+ }
+ }
+ }
+
+ if (!read_from_stdin) {
+ /* flcose() will flush the data and release the lock if
+ any acquired. */
+ fclose(fil_in);
+ }
+
+ /* Enabled for page type summary. */
+ if (page_type_summary) {
+ if (!read_from_stdin) {
+ fprintf(stdout, "\nFile::%s",filename);
+ print_summary(stdout);
+ } else {
+ print_summary(stderr);
+ }
+ }
+ }
- if (compressed)
- {
- printf("Table is compressed\n");
- printf("Key block size is %lu\n", physical_page_size);
- }
- else
- {
- printf("Table is uncompressed\n");
- printf("Page size is %lu\n", physical_page_size);
- }
+ if (is_log_enabled) {
+ fclose(log_file);
+ }
- pages= (ulint) (size / physical_page_size);
+ free(buf);
+ free(xdes);
+ my_end(0);
- if (just_count)
- {
- if (verbose)
- printf("Number of pages: ");
- printf("%lu\n", pages);
- goto ok;
- }
- else if (verbose)
- {
- printf("file %s = %llu bytes (%lu pages)...\n", filename, size, (ulong)pages);
- if (do_one_page)
- printf("InnoChecksum; checking page %lu\n", do_page);
- else
- printf("InnoChecksum; checking pages in range %lu to %lu\n", start_page, use_end_page ? end_page : (pages - 1));
- }
+ DBUG_RETURN(0);
-#ifdef UNIV_LINUX
- if (posix_fadvise(fileno(f), 0, 0, POSIX_FADV_SEQUENTIAL) ||
- posix_fadvise(fileno(f), 0, 0, POSIX_FADV_NOREUSE))
- {
- perror("posix_fadvise failed");
- }
-#endif
+err_exit:
+ if (buf) {
+ free(buf);
+ }
- /* seek to the necessary position */
- if (start_page)
- {
- fd= fileno(f);
- if (!fd)
- {
- perror("Error; Unable to obtain file descriptor number");
- goto error;
- }
-
- offset= (longlong)start_page * (longlong)physical_page_size;
-#ifdef _WIN32
- if (_lseeki64(fd, offset, SEEK_SET) != offset)
-#else
- if (lseek(fd, offset, SEEK_SET) != offset)
-#endif
- {
- perror("Error; Unable to seek to necessary offset");
- goto error;
- }
- }
+ if (xdes) {
+ free(xdes);
+ }
- /* main checksumming loop */
- ct= start_page;
- lastt= 0;
- while (!feof(f))
- {
- int page_ok = 1;
-
- bytes= fread(buf, 1, physical_page_size, f);
-
- if (!bytes && feof(f))
- goto ok;
-
- if (ferror(f))
- {
- fprintf(stderr, "Error reading %lu bytes", physical_page_size);
- perror(" ");
- goto error;
- }
-
- ulint page_type = mach_read_from_2(buf+FIL_PAGE_TYPE);
- ulint key_version = mach_read_from_4(buf + FIL_PAGE_FILE_FLUSH_LSN_OR_KEY_VERSION);
-
- if (key_version && page_type != FIL_PAGE_PAGE_COMPRESSED) {
- encrypted = true;
- } else {
- encrypted = false;
- }
-
- ulint comp_method = 0;
-
- if (encrypted) {
- comp_method = mach_read_from_2(buf+FIL_PAGE_DATA+FIL_PAGE_COMPRESSED_SIZE);
- } else {
- comp_method = mach_read_from_8(buf+FIL_PAGE_FILE_FLUSH_LSN_OR_KEY_VERSION);
- }
-
- ulint comp_size = mach_read_from_2(buf+FIL_PAGE_DATA);
- ib_uint32_t encryption_checksum = mach_read_from_4(buf+FIL_PAGE_FILE_FLUSH_LSN_OR_KEY_VERSION + 4);
-
-
- if (page_type == FIL_PAGE_PAGE_COMPRESSED) {
- /* Page compressed tables do not have any checksum */
- if (debug)
- fprintf(stderr, "Page %lu page compressed with method %s real_size %lu\n", ct,
- fil_get_compression_alg_name(comp_method), comp_size);
- page_ok = 1;
- } else if (compressed) {
- /* compressed pages */
- ulint crccsum = page_zip_calc_checksum(buf, physical_page_size, SRV_CHECKSUM_ALGORITHM_CRC32);
- ulint icsum = page_zip_calc_checksum(buf, physical_page_size, SRV_CHECKSUM_ALGORITHM_INNODB);
-
- if (debug) {
- if (key_version != 0) {
- fprintf(stderr,
- "Page %lu encrypted key_version %lu calculated = %lu; crc32 = %lu; recorded = %u\n",
- ct, key_version, icsum, crccsum, encryption_checksum);
- }
- }
+ if (!read_from_stdin && fil_in) {
+ fclose(fil_in);
+ }
- if (encrypted) {
- if (encryption_checksum != 0 && crccsum != encryption_checksum && icsum != encryption_checksum) {
- if (debug)
- fprintf(stderr, "page %lu: compressed: calculated = %lu; crc32 = %lu; recorded = %u\n",
- ct, icsum, crccsum, encryption_checksum);
- fprintf(stderr, "Fail; page %lu invalid (fails compressed page checksum).\n", ct);
- }
- } else {
- if (!page_zip_verify_checksum(buf, physical_page_size)) {
- fprintf(stderr, "Fail; page %lu invalid (fails compressed page checksum).\n", ct);
- if (!skip_corrupt)
- goto error;
- page_ok = 0;
- }
+ if (log_file) {
+ fclose(log_file);
}
- } else {
- if (key_version != 0) {
- /* Encrypted page */
- if (debug) {
- if (page_type == FIL_PAGE_PAGE_COMPRESSED_ENCRYPTED) {
- fprintf(stderr,
- "Page %lu page compressed with method %s real_size %lu and encrypted key_version %lu checksum %u\n",
- ct, fil_get_compression_alg_name(comp_method), comp_size, key_version, encryption_checksum);
- } else {
- fprintf(stderr,
- "Page %lu encrypted key_version %lu checksum %u\n",
- ct, key_version, encryption_checksum);
- }
- }
- }
-
- /* Page compressed tables do not contain FIL tailer */
- if (page_type != FIL_PAGE_PAGE_COMPRESSED_ENCRYPTED && page_type != FIL_PAGE_PAGE_COMPRESSED) {
- /* check the "stored log sequence numbers" */
- logseq= mach_read_from_4(buf + FIL_PAGE_LSN + 4);
- logseqfield= mach_read_from_4(buf + logical_page_size - FIL_PAGE_END_LSN_OLD_CHKSUM + 4);
- if (debug)
- printf("page %lu: log sequence number: first = %lu; second = %lu\n", ct, logseq, logseqfield);
- if (logseq != logseqfield)
- {
- fprintf(stderr, "Fail; page %lu invalid (fails log sequence number check)\n", ct);
- if (!skip_corrupt)
- goto error;
- page_ok = 0;
- }
- /* check old method of checksumming */
- oldcsum= buf_calc_page_old_checksum(buf);
- oldcsumfield= mach_read_from_4(buf + logical_page_size - FIL_PAGE_END_LSN_OLD_CHKSUM);
- if (debug)
- printf("page %lu: old style: calculated = %lu; recorded = %lu\n", ct, oldcsum, oldcsumfield);
- if (oldcsumfield != mach_read_from_4(buf + FIL_PAGE_LSN) && oldcsumfield != oldcsum)
- {
- fprintf(stderr, "Fail; page %lu invalid (fails old style checksum)\n", ct);
- if (!skip_corrupt)
- goto error;
- page_ok = 0;
- }
- }
-
- /* now check the new method */
- csum= buf_calc_page_new_checksum(buf);
- crc32= buf_calc_page_crc32(buf);
- csumfield= mach_read_from_4(buf + FIL_PAGE_SPACE_OR_CHKSUM);
-
- if (key_version)
- csumfield = encryption_checksum;
-
- if (debug)
- printf("page %lu: new style: calculated = %lu; crc32 = %lu; recorded = %lu\n",
- ct, csum, crc32, csumfield);
- if (csumfield != 0 && crc32 != csumfield && csum != csumfield)
- {
- fprintf(stderr, "Fail; page %lu invalid (fails innodb and crc32 checksum)\n", ct);
- if (!skip_corrupt)
- goto error;
- page_ok = 0;
- }
- }
- /* end if this was the last page we were supposed to check */
- if (use_end_page && (ct >= end_page))
- goto ok;
-
- if (per_page_details)
- {
- printf("page %ld ", ct);
- }
-
- /* do counter increase and progress printing */
- ct++;
-
- if (!page_ok)
- {
- if (per_page_details)
- {
- printf("BAD_CHECKSUM\n");
- }
- n_bad_checksum++;
- continue;
- }
-
- /* Can't parse compressed or/and encrypted pages */
- if (page_type != FIL_PAGE_PAGE_COMPRESSED && !encrypted) {
- parse_page(buf, xdes);
- }
-
- if (verbose)
- {
- if (ct % 64 == 0)
- {
- now= time(0);
- if (!lastt) lastt= now;
- if (now - lastt >= 1)
- {
- printf("page %lu okay: %.3f%% done\n", (ct - 1), (float) ct / pages * 100);
- lastt= now;
- }
- }
- }
- }
+ my_end(1);
-ok:
- if (!just_count)
- print_stats();
- free(big_xdes);
- free(big_buf);
- fclose(f);
- my_end(0);
- exit(0);
-
-error:
- free(big_xdes);
-error_big_buf:
- free(big_buf);
-error_f:
- fclose(f);
-error_out:
- my_end(0);
- exit(1);
+ DBUG_RETURN(1);
}