mirror of
https://github.com/MariaDB/server.git
synced 2025-01-23 15:24:16 +01:00
494e4b99a4
MySQL 5.7 introduced partial support for user-created shared tablespaces (for example, import and export are not supported). MariaDB Server does not support tablespaces at this point of time. Let us remove most InnoDB code and data structures that is related to shared tablespaces.
9558 lines
268 KiB
C++
9558 lines
268 KiB
C++
/*****************************************************************************
|
|
|
|
Copyright (c) 2007, 2016, Oracle and/or its affiliates.
|
|
Copyrigth (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 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, Suite 500, Boston, MA 02110-1335 USA
|
|
|
|
*****************************************************************************/
|
|
|
|
/**************************************************//**
|
|
@file handler/i_s.cc
|
|
InnoDB INFORMATION SCHEMA tables interface to MySQL.
|
|
|
|
Created July 18, 2007 Vasil Dimov
|
|
Modified Dec 29, 2014 Jan Lindström (Added sys_semaphore_waits)
|
|
*******************************************************/
|
|
|
|
#include "ha_prototypes.h"
|
|
#include <mysql_version.h>
|
|
#include <field.h>
|
|
#include "univ.i"
|
|
|
|
#include <sql_acl.h>
|
|
#include <sql_show.h>
|
|
#include <sql_time.h>
|
|
|
|
#include "i_s.h"
|
|
#include "btr0pcur.h"
|
|
#include "btr0types.h"
|
|
#include "dict0dict.h"
|
|
#include "dict0load.h"
|
|
#include "buf0buddy.h"
|
|
#include "buf0buf.h"
|
|
#include "ibuf0ibuf.h"
|
|
#include "dict0mem.h"
|
|
#include "dict0types.h"
|
|
#include "srv0start.h"
|
|
#include "trx0i_s.h"
|
|
#include "trx0trx.h"
|
|
#include "srv0mon.h"
|
|
#include "fut0fut.h"
|
|
#include "pars0pars.h"
|
|
#include "fts0types.h"
|
|
#include "fts0opt.h"
|
|
#include "fts0priv.h"
|
|
#include "btr0btr.h"
|
|
#include "page0zip.h"
|
|
#include "sync0arr.h"
|
|
#include "fil0fil.h"
|
|
#include "fil0crypt.h"
|
|
#include "fsp0sysspace.h"
|
|
#include "ut0new.h"
|
|
#include "dict0crea.h"
|
|
|
|
/** structure associates a name string with a file page type and/or buffer
|
|
page state. */
|
|
struct buf_page_desc_t{
|
|
const char* type_str; /*!< String explain the page
|
|
type/state */
|
|
ulint type_value; /*!< Page type or page state */
|
|
};
|
|
|
|
/** We also define I_S_PAGE_TYPE_INDEX as the Index Page's position
|
|
in i_s_page_type[] array */
|
|
#define I_S_PAGE_TYPE_INDEX 1
|
|
|
|
/** Any unassigned FIL_PAGE_TYPE will be treated as unknown. */
|
|
#define I_S_PAGE_TYPE_UNKNOWN FIL_PAGE_TYPE_UNKNOWN
|
|
|
|
/** R-tree index page */
|
|
#define I_S_PAGE_TYPE_RTREE (FIL_PAGE_TYPE_LAST + 1)
|
|
|
|
/** Change buffer B-tree page */
|
|
#define I_S_PAGE_TYPE_IBUF (FIL_PAGE_TYPE_LAST + 2)
|
|
|
|
#define I_S_PAGE_TYPE_LAST I_S_PAGE_TYPE_IBUF
|
|
|
|
#define I_S_PAGE_TYPE_BITS 4
|
|
|
|
/* Check if we can hold all page types */
|
|
#if I_S_PAGE_TYPE_LAST >= 1 << I_S_PAGE_TYPE_BITS
|
|
# error i_s_page_type[] is too large
|
|
#endif
|
|
|
|
/** Name string for File Page Types */
|
|
static buf_page_desc_t i_s_page_type[] = {
|
|
{"ALLOCATED", FIL_PAGE_TYPE_ALLOCATED},
|
|
{"INDEX", FIL_PAGE_INDEX},
|
|
{"UNDO_LOG", FIL_PAGE_UNDO_LOG},
|
|
{"INODE", FIL_PAGE_INODE},
|
|
{"IBUF_FREE_LIST", FIL_PAGE_IBUF_FREE_LIST},
|
|
{"IBUF_BITMAP", FIL_PAGE_IBUF_BITMAP},
|
|
{"SYSTEM", FIL_PAGE_TYPE_SYS},
|
|
{"TRX_SYSTEM", FIL_PAGE_TYPE_TRX_SYS},
|
|
{"FILE_SPACE_HEADER", FIL_PAGE_TYPE_FSP_HDR},
|
|
{"EXTENT_DESCRIPTOR", FIL_PAGE_TYPE_XDES},
|
|
{"BLOB", FIL_PAGE_TYPE_BLOB},
|
|
{"COMPRESSED_BLOB", FIL_PAGE_TYPE_ZBLOB},
|
|
{"COMPRESSED_BLOB2", FIL_PAGE_TYPE_ZBLOB2},
|
|
{"UNKNOWN", I_S_PAGE_TYPE_UNKNOWN},
|
|
{"RTREE_INDEX", I_S_PAGE_TYPE_RTREE},
|
|
{"IBUF_INDEX", I_S_PAGE_TYPE_IBUF},
|
|
{"PAGE COMPRESSED", FIL_PAGE_PAGE_COMPRESSED},
|
|
{"PAGE COMPRESSED AND ENCRYPTED", FIL_PAGE_PAGE_COMPRESSED_ENCRYPTED},
|
|
};
|
|
|
|
/** This structure defines information we will fetch from pages
|
|
currently cached in the buffer pool. It will be used to populate
|
|
table INFORMATION_SCHEMA.INNODB_BUFFER_PAGE */
|
|
struct buf_page_info_t{
|
|
ulint block_id; /*!< Buffer Pool block ID */
|
|
unsigned space_id:32; /*!< Tablespace ID */
|
|
unsigned page_num:32; /*!< Page number/offset */
|
|
unsigned access_time:32; /*!< Time of first access */
|
|
unsigned pool_id:MAX_BUFFER_POOLS_BITS;
|
|
/*!< Buffer Pool ID. Must be less than
|
|
MAX_BUFFER_POOLS */
|
|
unsigned flush_type:2; /*!< Flush type */
|
|
unsigned io_fix:2; /*!< type of pending I/O operation */
|
|
unsigned fix_count:19; /*!< Count of how manyfold this block
|
|
is bufferfixed */
|
|
unsigned hashed:1; /*!< Whether hash index has been
|
|
built on this page */
|
|
unsigned is_old:1; /*!< TRUE if the block is in the old
|
|
blocks in buf_pool->LRU_old */
|
|
unsigned freed_page_clock:31; /*!< the value of
|
|
buf_pool->freed_page_clock */
|
|
unsigned zip_ssize:PAGE_ZIP_SSIZE_BITS;
|
|
/*!< Compressed page size */
|
|
unsigned page_state:BUF_PAGE_STATE_BITS; /*!< Page state */
|
|
unsigned page_type:I_S_PAGE_TYPE_BITS; /*!< Page type */
|
|
unsigned num_recs:UNIV_PAGE_SIZE_SHIFT_MAX-2;
|
|
/*!< Number of records on Page */
|
|
unsigned data_size:UNIV_PAGE_SIZE_SHIFT_MAX;
|
|
/*!< Sum of the sizes of the records */
|
|
lsn_t newest_mod; /*!< Log sequence number of
|
|
the youngest modification */
|
|
lsn_t oldest_mod; /*!< Log sequence number of
|
|
the oldest modification */
|
|
index_id_t index_id; /*!< Index ID if a index page */
|
|
};
|
|
|
|
/*
|
|
Use the following types mapping:
|
|
|
|
C type ST_FIELD_INFO::field_type
|
|
---------------------------------
|
|
long MYSQL_TYPE_LONGLONG
|
|
(field_length=MY_INT64_NUM_DECIMAL_DIGITS)
|
|
|
|
long unsigned MYSQL_TYPE_LONGLONG
|
|
(field_length=MY_INT64_NUM_DECIMAL_DIGITS, field_flags=MY_I_S_UNSIGNED)
|
|
|
|
char* MYSQL_TYPE_STRING
|
|
(field_length=n)
|
|
|
|
float MYSQL_TYPE_FLOAT
|
|
(field_length=0 is ignored)
|
|
|
|
void* MYSQL_TYPE_LONGLONG
|
|
(field_length=MY_INT64_NUM_DECIMAL_DIGITS, field_flags=MY_I_S_UNSIGNED)
|
|
|
|
boolean (if else) MYSQL_TYPE_LONG
|
|
(field_length=1)
|
|
|
|
time_t MYSQL_TYPE_DATETIME
|
|
(field_length=0 ignored)
|
|
---------------------------------
|
|
*/
|
|
|
|
/** Implemented on sync0arr.cc */
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.INNODB_SYS_SEMAPHORE_WAITS table.
|
|
Loop through each item on sync array, and extract the column
|
|
information and fill the INFORMATION_SCHEMA.INNODB_SYS_SEMAPHORE_WAITS table.
|
|
@return 0 on success */
|
|
UNIV_INTERN
|
|
int
|
|
sync_arr_fill_sys_semphore_waits_table(
|
|
/*===================================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ); /*!< in: condition (not used) */
|
|
|
|
/*******************************************************************//**
|
|
Common function to fill any of the dynamic tables:
|
|
INFORMATION_SCHEMA.innodb_trx
|
|
INFORMATION_SCHEMA.innodb_locks
|
|
INFORMATION_SCHEMA.innodb_lock_waits
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
trx_i_s_common_fill_table(
|
|
/*======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ); /*!< in: condition (not used) */
|
|
|
|
/*******************************************************************//**
|
|
Unbind a dynamic INFORMATION_SCHEMA table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_common_deinit(
|
|
/*==============*/
|
|
void* p); /*!< in/out: table schema object */
|
|
/*******************************************************************//**
|
|
Auxiliary function to store time_t value in MYSQL_TYPE_DATETIME
|
|
field.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
field_store_time_t(
|
|
/*===============*/
|
|
Field* field, /*!< in/out: target field for storage */
|
|
time_t time) /*!< in: value to store */
|
|
{
|
|
MYSQL_TIME my_time;
|
|
struct tm tm_time;
|
|
|
|
if (time) {
|
|
#if 0
|
|
/* use this if you are sure that `variables' and `time_zone'
|
|
are always initialized */
|
|
thd->variables.time_zone->gmt_sec_to_TIME(
|
|
&my_time, (my_time_t) time);
|
|
#else
|
|
localtime_r(&time, &tm_time);
|
|
localtime_to_TIME(&my_time, &tm_time);
|
|
my_time.time_type = MYSQL_TIMESTAMP_DATETIME;
|
|
#endif
|
|
} else {
|
|
memset(&my_time, 0, sizeof(my_time));
|
|
}
|
|
|
|
/* JAN: TODO: MySQL 5.7
|
|
return(field->store_time(&my_time, MYSQL_TIMESTAMP_DATETIME));
|
|
*/
|
|
return(field->store_time(&my_time));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Auxiliary function to store char* value in MYSQL_TYPE_STRING field.
|
|
@return 0 on success */
|
|
int
|
|
field_store_string(
|
|
/*===============*/
|
|
Field* field, /*!< in/out: target field for storage */
|
|
const char* str) /*!< in: NUL-terminated utf-8 string,
|
|
or NULL */
|
|
{
|
|
int ret;
|
|
|
|
if (str != NULL) {
|
|
|
|
ret = field->store(str, static_cast<uint>(strlen(str)),
|
|
system_charset_info);
|
|
field->set_notnull();
|
|
} else {
|
|
|
|
ret = 0; /* success */
|
|
field->set_null();
|
|
}
|
|
|
|
return(ret);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Store the name of an index in a MYSQL_TYPE_VARCHAR field.
|
|
Handles the names of incomplete secondary indexes.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
field_store_index_name(
|
|
/*===================*/
|
|
Field* field, /*!< in/out: target field for
|
|
storage */
|
|
const char* index_name) /*!< in: NUL-terminated utf-8
|
|
index name, possibly starting with
|
|
TEMP_INDEX_PREFIX */
|
|
{
|
|
int ret;
|
|
|
|
ut_ad(index_name != NULL);
|
|
ut_ad(field->real_type() == MYSQL_TYPE_VARCHAR);
|
|
|
|
/* Since TEMP_INDEX_PREFIX is not a valid UTF8, we need to convert
|
|
it to something else. */
|
|
if (*index_name == *TEMP_INDEX_PREFIX_STR) {
|
|
char buf[NAME_LEN + 1];
|
|
buf[0] = '?';
|
|
memcpy(buf + 1, index_name + 1, strlen(index_name));
|
|
ret = field->store(
|
|
buf, static_cast<uint>(strlen(buf)),
|
|
system_charset_info);
|
|
} else {
|
|
ret = field->store(
|
|
index_name, static_cast<uint>(strlen(index_name)),
|
|
system_charset_info);
|
|
}
|
|
|
|
field->set_notnull();
|
|
|
|
return(ret);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Auxiliary function to store ulint value in MYSQL_TYPE_LONGLONG field.
|
|
If the value is ULINT_UNDEFINED then the field is set to NULL.
|
|
@return 0 on success */
|
|
int
|
|
field_store_ulint(
|
|
/*==============*/
|
|
Field* field, /*!< in/out: target field for storage */
|
|
ulint n) /*!< in: value to store */
|
|
{
|
|
int ret;
|
|
|
|
if (n != ULINT_UNDEFINED) {
|
|
|
|
ret = field->store(n, true);
|
|
field->set_notnull();
|
|
} else {
|
|
|
|
ret = 0; /* success */
|
|
field->set_null();
|
|
}
|
|
|
|
return(ret);
|
|
}
|
|
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.innodb_trx */
|
|
static ST_FIELD_INFO innodb_trx_fields_info[] =
|
|
{
|
|
#define IDX_TRX_ID 0
|
|
{STRUCT_FLD(field_name, "trx_id"),
|
|
STRUCT_FLD(field_length, TRX_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_STATE 1
|
|
{STRUCT_FLD(field_name, "trx_state"),
|
|
STRUCT_FLD(field_length, TRX_QUE_STATE_STR_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_STARTED 2
|
|
{STRUCT_FLD(field_name, "trx_started"),
|
|
STRUCT_FLD(field_length, 0),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_DATETIME),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_REQUESTED_LOCK_ID 3
|
|
{STRUCT_FLD(field_name, "trx_requested_lock_id"),
|
|
STRUCT_FLD(field_length, TRX_I_S_LOCK_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_WAIT_STARTED 4
|
|
{STRUCT_FLD(field_name, "trx_wait_started"),
|
|
STRUCT_FLD(field_length, 0),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_DATETIME),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_WEIGHT 5
|
|
{STRUCT_FLD(field_name, "trx_weight"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_MYSQL_THREAD_ID 6
|
|
{STRUCT_FLD(field_name, "trx_mysql_thread_id"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_QUERY 7
|
|
{STRUCT_FLD(field_name, "trx_query"),
|
|
STRUCT_FLD(field_length, TRX_I_S_TRX_QUERY_MAX_LEN),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_OPERATION_STATE 8
|
|
{STRUCT_FLD(field_name, "trx_operation_state"),
|
|
STRUCT_FLD(field_length, TRX_I_S_TRX_OP_STATE_MAX_LEN),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_TABLES_IN_USE 9
|
|
{STRUCT_FLD(field_name, "trx_tables_in_use"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_TABLES_LOCKED 10
|
|
{STRUCT_FLD(field_name, "trx_tables_locked"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_LOCK_STRUCTS 11
|
|
{STRUCT_FLD(field_name, "trx_lock_structs"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_LOCK_MEMORY_BYTES 12
|
|
{STRUCT_FLD(field_name, "trx_lock_memory_bytes"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_ROWS_LOCKED 13
|
|
{STRUCT_FLD(field_name, "trx_rows_locked"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_ROWS_MODIFIED 14
|
|
{STRUCT_FLD(field_name, "trx_rows_modified"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_CONNCURRENCY_TICKETS 15
|
|
{STRUCT_FLD(field_name, "trx_concurrency_tickets"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_ISOLATION_LEVEL 16
|
|
{STRUCT_FLD(field_name, "trx_isolation_level"),
|
|
STRUCT_FLD(field_length, TRX_I_S_TRX_ISOLATION_LEVEL_MAX_LEN),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_UNIQUE_CHECKS 17
|
|
{STRUCT_FLD(field_name, "trx_unique_checks"),
|
|
STRUCT_FLD(field_length, 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 1),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_FOREIGN_KEY_CHECKS 18
|
|
{STRUCT_FLD(field_name, "trx_foreign_key_checks"),
|
|
STRUCT_FLD(field_length, 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 1),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_LAST_FOREIGN_KEY_ERROR 19
|
|
{STRUCT_FLD(field_name, "trx_last_foreign_key_error"),
|
|
STRUCT_FLD(field_length, TRX_I_S_TRX_FK_ERROR_MAX_LEN),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_ADAPTIVE_HASH_LATCHED 20
|
|
{STRUCT_FLD(field_name, "trx_adaptive_hash_latched"),
|
|
STRUCT_FLD(field_length, 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_ADAPTIVE_HASH_TIMEOUT 21
|
|
{STRUCT_FLD(field_name, "trx_adaptive_hash_timeout"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_READ_ONLY 22
|
|
{STRUCT_FLD(field_name, "trx_is_read_only"),
|
|
STRUCT_FLD(field_length, 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TRX_AUTOCOMMIT_NON_LOCKING 23
|
|
{STRUCT_FLD(field_name, "trx_autocommit_non_locking"),
|
|
STRUCT_FLD(field_length, 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Read data from cache buffer and fill the INFORMATION_SCHEMA.innodb_trx
|
|
table with it.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
fill_innodb_trx_from_cache(
|
|
/*=======================*/
|
|
trx_i_s_cache_t* cache, /*!< in: cache to read from */
|
|
THD* thd, /*!< in: used to call
|
|
schema_table_store_record() */
|
|
TABLE* table) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
ulint rows_num;
|
|
char lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
|
|
ulint i;
|
|
|
|
DBUG_ENTER("fill_innodb_trx_from_cache");
|
|
|
|
fields = table->field;
|
|
|
|
rows_num = trx_i_s_cache_get_rows_used(cache,
|
|
I_S_INNODB_TRX);
|
|
|
|
for (i = 0; i < rows_num; i++) {
|
|
|
|
i_s_trx_row_t* row;
|
|
char trx_id[TRX_ID_MAX_LEN + 1];
|
|
|
|
row = (i_s_trx_row_t*)
|
|
trx_i_s_cache_get_nth_row(
|
|
cache, I_S_INNODB_TRX, i);
|
|
|
|
/* trx_id */
|
|
ut_snprintf(trx_id, sizeof(trx_id), TRX_ID_FMT, row->trx_id);
|
|
OK(field_store_string(fields[IDX_TRX_ID], trx_id));
|
|
|
|
/* trx_state */
|
|
OK(field_store_string(fields[IDX_TRX_STATE],
|
|
row->trx_state));
|
|
|
|
/* trx_started */
|
|
OK(field_store_time_t(fields[IDX_TRX_STARTED],
|
|
(time_t) row->trx_started));
|
|
|
|
/* trx_requested_lock_id */
|
|
/* trx_wait_started */
|
|
if (row->trx_wait_started != 0) {
|
|
|
|
OK(field_store_string(
|
|
fields[IDX_TRX_REQUESTED_LOCK_ID],
|
|
trx_i_s_create_lock_id(
|
|
row->requested_lock_row,
|
|
lock_id, sizeof(lock_id))));
|
|
/* field_store_string() sets it no notnull */
|
|
|
|
OK(field_store_time_t(
|
|
fields[IDX_TRX_WAIT_STARTED],
|
|
(time_t) row->trx_wait_started));
|
|
fields[IDX_TRX_WAIT_STARTED]->set_notnull();
|
|
} else {
|
|
|
|
fields[IDX_TRX_REQUESTED_LOCK_ID]->set_null();
|
|
fields[IDX_TRX_WAIT_STARTED]->set_null();
|
|
}
|
|
|
|
/* trx_weight */
|
|
OK(fields[IDX_TRX_WEIGHT]->store(row->trx_weight, true));
|
|
|
|
/* trx_mysql_thread_id */
|
|
OK(fields[IDX_TRX_MYSQL_THREAD_ID]->store(
|
|
row->trx_mysql_thread_id, true));
|
|
|
|
/* trx_query */
|
|
if (row->trx_query) {
|
|
/* store will do appropriate character set
|
|
conversion check */
|
|
fields[IDX_TRX_QUERY]->store(
|
|
row->trx_query,
|
|
static_cast<uint>(strlen(row->trx_query)),
|
|
row->trx_query_cs);
|
|
fields[IDX_TRX_QUERY]->set_notnull();
|
|
} else {
|
|
fields[IDX_TRX_QUERY]->set_null();
|
|
}
|
|
|
|
/* trx_operation_state */
|
|
OK(field_store_string(fields[IDX_TRX_OPERATION_STATE],
|
|
row->trx_operation_state));
|
|
|
|
/* trx_tables_in_use */
|
|
OK(fields[IDX_TRX_TABLES_IN_USE]->store(
|
|
row->trx_tables_in_use, true));
|
|
|
|
/* trx_tables_locked */
|
|
OK(fields[IDX_TRX_TABLES_LOCKED]->store(
|
|
row->trx_tables_locked, true));
|
|
|
|
/* trx_lock_structs */
|
|
OK(fields[IDX_TRX_LOCK_STRUCTS]->store(
|
|
row->trx_lock_structs, true));
|
|
|
|
/* trx_lock_memory_bytes */
|
|
OK(fields[IDX_TRX_LOCK_MEMORY_BYTES]->store(
|
|
row->trx_lock_memory_bytes, true));
|
|
|
|
/* trx_rows_locked */
|
|
OK(fields[IDX_TRX_ROWS_LOCKED]->store(
|
|
row->trx_rows_locked, true));
|
|
|
|
/* trx_rows_modified */
|
|
OK(fields[IDX_TRX_ROWS_MODIFIED]->store(
|
|
row->trx_rows_modified, true));
|
|
|
|
/* trx_concurrency_tickets */
|
|
OK(fields[IDX_TRX_CONNCURRENCY_TICKETS]->store(
|
|
row->trx_concurrency_tickets, true));
|
|
|
|
/* trx_isolation_level */
|
|
OK(field_store_string(fields[IDX_TRX_ISOLATION_LEVEL],
|
|
row->trx_isolation_level));
|
|
|
|
/* trx_unique_checks */
|
|
OK(fields[IDX_TRX_UNIQUE_CHECKS]->store(
|
|
row->trx_unique_checks, true));
|
|
|
|
/* trx_foreign_key_checks */
|
|
OK(fields[IDX_TRX_FOREIGN_KEY_CHECKS]->store(
|
|
row->trx_foreign_key_checks, true));
|
|
|
|
/* trx_last_foreign_key_error */
|
|
OK(field_store_string(fields[IDX_TRX_LAST_FOREIGN_KEY_ERROR],
|
|
row->trx_foreign_key_error));
|
|
|
|
/* trx_adaptive_hash_latched */
|
|
OK(fields[IDX_TRX_ADAPTIVE_HASH_LATCHED]->store(
|
|
row->trx_has_search_latch, true));
|
|
|
|
/* trx_is_read_only*/
|
|
OK(fields[IDX_TRX_READ_ONLY]->store(
|
|
row->trx_is_read_only, true));
|
|
|
|
/* trx_is_autocommit_non_locking */
|
|
OK(fields[IDX_TRX_AUTOCOMMIT_NON_LOCKING]->store(
|
|
(longlong) row->trx_is_autocommit_non_locking,
|
|
true));
|
|
|
|
OK(schema_table_store_record(thd, table));
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_trx
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_trx_init(
|
|
/*============*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_trx_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_trx_fields_info;
|
|
schema->fill_table = trx_i_s_common_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
static struct st_mysql_information_schema i_s_info =
|
|
{
|
|
MYSQL_INFORMATION_SCHEMA_INTERFACE_VERSION
|
|
};
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_trx =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_TRX"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB transactions"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_trx_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.innodb_locks */
|
|
static ST_FIELD_INFO innodb_locks_fields_info[] =
|
|
{
|
|
#define IDX_LOCK_ID 0
|
|
{STRUCT_FLD(field_name, "lock_id"),
|
|
STRUCT_FLD(field_length, TRX_I_S_LOCK_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_TRX_ID 1
|
|
{STRUCT_FLD(field_name, "lock_trx_id"),
|
|
STRUCT_FLD(field_length, TRX_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_MODE 2
|
|
{STRUCT_FLD(field_name, "lock_mode"),
|
|
/* S[,GAP] X[,GAP] IS[,GAP] IX[,GAP] AUTO_INC UNKNOWN */
|
|
STRUCT_FLD(field_length, 32),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_TYPE 3
|
|
{STRUCT_FLD(field_name, "lock_type"),
|
|
STRUCT_FLD(field_length, 32 /* RECORD|TABLE|UNKNOWN */),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_TABLE 4
|
|
{STRUCT_FLD(field_name, "lock_table"),
|
|
STRUCT_FLD(field_length, 1024),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_INDEX 5
|
|
{STRUCT_FLD(field_name, "lock_index"),
|
|
STRUCT_FLD(field_length, 1024),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_SPACE 6
|
|
{STRUCT_FLD(field_name, "lock_space"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED | MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_PAGE 7
|
|
{STRUCT_FLD(field_name, "lock_page"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED | MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_REC 8
|
|
{STRUCT_FLD(field_name, "lock_rec"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED | MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_LOCK_DATA 9
|
|
{STRUCT_FLD(field_name, "lock_data"),
|
|
STRUCT_FLD(field_length, TRX_I_S_LOCK_DATA_MAX_LEN),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Read data from cache buffer and fill the INFORMATION_SCHEMA.innodb_locks
|
|
table with it.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
fill_innodb_locks_from_cache(
|
|
/*=========================*/
|
|
trx_i_s_cache_t* cache, /*!< in: cache to read from */
|
|
THD* thd, /*!< in: MySQL client connection */
|
|
TABLE* table) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
ulint rows_num;
|
|
char lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
|
|
ulint i;
|
|
|
|
DBUG_ENTER("fill_innodb_locks_from_cache");
|
|
|
|
fields = table->field;
|
|
|
|
rows_num = trx_i_s_cache_get_rows_used(cache,
|
|
I_S_INNODB_LOCKS);
|
|
|
|
for (i = 0; i < rows_num; i++) {
|
|
|
|
i_s_locks_row_t* row;
|
|
char buf[MAX_FULL_NAME_LEN + 1];
|
|
const char* bufend;
|
|
|
|
char lock_trx_id[TRX_ID_MAX_LEN + 1];
|
|
|
|
row = (i_s_locks_row_t*)
|
|
trx_i_s_cache_get_nth_row(
|
|
cache, I_S_INNODB_LOCKS, i);
|
|
|
|
/* lock_id */
|
|
trx_i_s_create_lock_id(row, lock_id, sizeof(lock_id));
|
|
OK(field_store_string(fields[IDX_LOCK_ID],
|
|
lock_id));
|
|
|
|
/* lock_trx_id */
|
|
ut_snprintf(lock_trx_id, sizeof(lock_trx_id),
|
|
TRX_ID_FMT, row->lock_trx_id);
|
|
OK(field_store_string(fields[IDX_LOCK_TRX_ID], lock_trx_id));
|
|
|
|
/* lock_mode */
|
|
OK(field_store_string(fields[IDX_LOCK_MODE],
|
|
row->lock_mode));
|
|
|
|
/* lock_type */
|
|
OK(field_store_string(fields[IDX_LOCK_TYPE],
|
|
row->lock_type));
|
|
|
|
/* lock_table */
|
|
bufend = innobase_convert_name(buf, sizeof(buf),
|
|
row->lock_table,
|
|
strlen(row->lock_table),
|
|
thd);
|
|
OK(fields[IDX_LOCK_TABLE]->store(
|
|
buf, static_cast<size_t>(bufend - buf),
|
|
system_charset_info));
|
|
|
|
/* lock_index */
|
|
if (row->lock_index != NULL) {
|
|
OK(field_store_index_name(fields[IDX_LOCK_INDEX],
|
|
row->lock_index));
|
|
} else {
|
|
fields[IDX_LOCK_INDEX]->set_null();
|
|
}
|
|
|
|
/* lock_space */
|
|
OK(field_store_ulint(fields[IDX_LOCK_SPACE],
|
|
row->lock_space));
|
|
|
|
/* lock_page */
|
|
OK(field_store_ulint(fields[IDX_LOCK_PAGE],
|
|
row->lock_page));
|
|
|
|
/* lock_rec */
|
|
OK(field_store_ulint(fields[IDX_LOCK_REC],
|
|
row->lock_rec));
|
|
|
|
/* lock_data */
|
|
OK(field_store_string(fields[IDX_LOCK_DATA],
|
|
row->lock_data));
|
|
|
|
OK(schema_table_store_record(thd, table));
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_locks
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_locks_init(
|
|
/*==============*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_locks_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_locks_fields_info;
|
|
schema->fill_table = trx_i_s_common_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_locks =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_LOCKS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB conflicting locks"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_locks_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.innodb_lock_waits */
|
|
static ST_FIELD_INFO innodb_lock_waits_fields_info[] =
|
|
{
|
|
#define IDX_REQUESTING_TRX_ID 0
|
|
{STRUCT_FLD(field_name, "requesting_trx_id"),
|
|
STRUCT_FLD(field_length, TRX_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_REQUESTED_LOCK_ID 1
|
|
{STRUCT_FLD(field_name, "requested_lock_id"),
|
|
STRUCT_FLD(field_length, TRX_I_S_LOCK_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BLOCKING_TRX_ID 2
|
|
{STRUCT_FLD(field_name, "blocking_trx_id"),
|
|
STRUCT_FLD(field_length, TRX_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BLOCKING_LOCK_ID 3
|
|
{STRUCT_FLD(field_name, "blocking_lock_id"),
|
|
STRUCT_FLD(field_length, TRX_I_S_LOCK_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Read data from cache buffer and fill the
|
|
INFORMATION_SCHEMA.innodb_lock_waits table with it.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
fill_innodb_lock_waits_from_cache(
|
|
/*==============================*/
|
|
trx_i_s_cache_t* cache, /*!< in: cache to read from */
|
|
THD* thd, /*!< in: used to call
|
|
schema_table_store_record() */
|
|
TABLE* table) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
ulint rows_num;
|
|
char requested_lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
|
|
char blocking_lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
|
|
ulint i;
|
|
|
|
DBUG_ENTER("fill_innodb_lock_waits_from_cache");
|
|
|
|
fields = table->field;
|
|
|
|
rows_num = trx_i_s_cache_get_rows_used(cache,
|
|
I_S_INNODB_LOCK_WAITS);
|
|
|
|
for (i = 0; i < rows_num; i++) {
|
|
|
|
i_s_lock_waits_row_t* row;
|
|
|
|
char requesting_trx_id[TRX_ID_MAX_LEN + 1];
|
|
char blocking_trx_id[TRX_ID_MAX_LEN + 1];
|
|
|
|
row = (i_s_lock_waits_row_t*)
|
|
trx_i_s_cache_get_nth_row(
|
|
cache, I_S_INNODB_LOCK_WAITS, i);
|
|
|
|
/* requesting_trx_id */
|
|
ut_snprintf(requesting_trx_id, sizeof(requesting_trx_id),
|
|
TRX_ID_FMT, row->requested_lock_row->lock_trx_id);
|
|
OK(field_store_string(fields[IDX_REQUESTING_TRX_ID],
|
|
requesting_trx_id));
|
|
|
|
/* requested_lock_id */
|
|
OK(field_store_string(
|
|
fields[IDX_REQUESTED_LOCK_ID],
|
|
trx_i_s_create_lock_id(
|
|
row->requested_lock_row,
|
|
requested_lock_id,
|
|
sizeof(requested_lock_id))));
|
|
|
|
/* blocking_trx_id */
|
|
ut_snprintf(blocking_trx_id, sizeof(blocking_trx_id),
|
|
TRX_ID_FMT, row->blocking_lock_row->lock_trx_id);
|
|
OK(field_store_string(fields[IDX_BLOCKING_TRX_ID],
|
|
blocking_trx_id));
|
|
|
|
/* blocking_lock_id */
|
|
OK(field_store_string(
|
|
fields[IDX_BLOCKING_LOCK_ID],
|
|
trx_i_s_create_lock_id(
|
|
row->blocking_lock_row,
|
|
blocking_lock_id,
|
|
sizeof(blocking_lock_id))));
|
|
|
|
OK(schema_table_store_record(thd, table));
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_lock_waits
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_lock_waits_init(
|
|
/*===================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_lock_waits_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_lock_waits_fields_info;
|
|
schema->fill_table = trx_i_s_common_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_lock_waits =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_LOCK_WAITS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB which lock is blocking which"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_lock_waits_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Common function to fill any of the dynamic tables:
|
|
INFORMATION_SCHEMA.innodb_trx
|
|
INFORMATION_SCHEMA.innodb_locks
|
|
INFORMATION_SCHEMA.innodb_lock_waits
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
trx_i_s_common_fill_table(
|
|
/*======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
const char* table_name;
|
|
int ret;
|
|
trx_i_s_cache_t* cache;
|
|
|
|
DBUG_ENTER("trx_i_s_common_fill_table");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* minimize the number of places where global variables are
|
|
referenced */
|
|
cache = trx_i_s_cache;
|
|
|
|
/* which table we have to fill? */
|
|
table_name = tables->schema_table_name;
|
|
/* or table_name = tables->schema_table->table_name; */
|
|
|
|
RETURN_IF_INNODB_NOT_STARTED(table_name);
|
|
|
|
/* update the cache */
|
|
trx_i_s_cache_start_write(cache);
|
|
trx_i_s_possibly_fetch_data_into_cache(cache);
|
|
trx_i_s_cache_end_write(cache);
|
|
|
|
if (trx_i_s_cache_is_truncated(cache)) {
|
|
|
|
ib::warn() << "Data in " << table_name << " truncated due to"
|
|
" memory limit of " << TRX_I_S_MEM_LIMIT << " bytes";
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
trx_i_s_cache_start_read(cache);
|
|
|
|
if (innobase_strcasecmp(table_name, "innodb_trx") == 0) {
|
|
|
|
if (fill_innodb_trx_from_cache(
|
|
cache, thd, tables->table) != 0) {
|
|
|
|
ret = 1;
|
|
}
|
|
|
|
} else if (innobase_strcasecmp(table_name, "innodb_locks") == 0) {
|
|
|
|
if (fill_innodb_locks_from_cache(
|
|
cache, thd, tables->table) != 0) {
|
|
|
|
ret = 1;
|
|
}
|
|
|
|
} else if (innobase_strcasecmp(table_name, "innodb_lock_waits") == 0) {
|
|
|
|
if (fill_innodb_lock_waits_from_cache(
|
|
cache, thd, tables->table) != 0) {
|
|
|
|
ret = 1;
|
|
}
|
|
|
|
} else {
|
|
ib::error() << "trx_i_s_common_fill_table() was"
|
|
" called to fill unknown table: " << table_name << "."
|
|
" This function only knows how to fill"
|
|
" innodb_trx, innodb_locks and"
|
|
" innodb_lock_waits tables.";
|
|
|
|
ret = 1;
|
|
}
|
|
|
|
trx_i_s_cache_end_read(cache);
|
|
|
|
#if 0
|
|
DBUG_RETURN(ret);
|
|
#else
|
|
/* if this function returns something else than 0 then a
|
|
deadlock occurs between the mysqld server and mysql client,
|
|
see http://bugs.mysql.com/29900 ; when that bug is resolved
|
|
we can enable the DBUG_RETURN(ret) above */
|
|
ret++; // silence a gcc46 warning
|
|
DBUG_RETURN(0);
|
|
#endif
|
|
}
|
|
|
|
/* Fields of the dynamic table information_schema.innodb_cmp. */
|
|
static ST_FIELD_INFO i_s_cmp_fields_info[] =
|
|
{
|
|
{STRUCT_FLD(field_name, "page_size"),
|
|
STRUCT_FLD(field_length, 5),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Compressed Page Size"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "compress_ops"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Total Number of Compressions"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "compress_ops_ok"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Total Number of"
|
|
" Successful Compressions"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "compress_time"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Total Duration of Compressions,"
|
|
" in Seconds"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "uncompress_ops"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Total Number of Decompressions"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "uncompress_time"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Total Duration of Decompressions,"
|
|
" in Seconds"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmp or
|
|
innodb_cmp_reset.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmp_fill_low(
|
|
/*=============*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* , /*!< in: condition (ignored) */
|
|
ibool reset) /*!< in: TRUE=reset cumulated counts */
|
|
{
|
|
TABLE* table = (TABLE*) tables->table;
|
|
int status = 0;
|
|
|
|
DBUG_ENTER("i_s_cmp_fill_low");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
for (uint i = 0; i < PAGE_ZIP_SSIZE_MAX; i++) {
|
|
page_zip_stat_t* zip_stat = &page_zip_stat[i];
|
|
|
|
table->field[0]->store(UNIV_ZIP_SIZE_MIN << i);
|
|
|
|
/* The cumulated counts are not protected by any
|
|
mutex. Thus, some operation in page0zip.cc could
|
|
increment a counter between the time we read it and
|
|
clear it. We could introduce mutex protection, but it
|
|
could cause a measureable performance hit in
|
|
page0zip.cc. */
|
|
table->field[1]->store(zip_stat->compressed, true);
|
|
table->field[2]->store(zip_stat->compressed_ok, true);
|
|
table->field[3]->store(zip_stat->compressed_usec / 1000000, true);
|
|
table->field[4]->store(zip_stat->decompressed, true);
|
|
table->field[5]->store(zip_stat->decompressed_usec / 1000000, true);
|
|
|
|
if (reset) {
|
|
memset(zip_stat, 0, sizeof *zip_stat);
|
|
}
|
|
|
|
if (schema_table_store_record(thd, table)) {
|
|
status = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmp.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmp_fill(
|
|
/*=========*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* cond) /*!< in: condition (ignored) */
|
|
{
|
|
return(i_s_cmp_fill_low(thd, tables, cond, FALSE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmp_reset.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmp_reset_fill(
|
|
/*===============*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* cond) /*!< in: condition (ignored) */
|
|
{
|
|
return(i_s_cmp_fill_low(thd, tables, cond, TRUE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table information_schema.innodb_cmp.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_cmp_init(
|
|
/*=========*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_cmp_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_cmp_fields_info;
|
|
schema->fill_table = i_s_cmp_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table information_schema.innodb_cmp_reset.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_cmp_reset_init(
|
|
/*===============*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_cmp_reset_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_cmp_fields_info;
|
|
schema->fill_table = i_s_cmp_reset_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_cmp =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_CMP"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "Statistics for the InnoDB compression"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_cmp_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_cmp_reset =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_CMP_RESET"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "Statistics for the InnoDB compression;"
|
|
" reset cumulated counts"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_cmp_reset_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic tables
|
|
information_schema.innodb_cmp_per_index and
|
|
information_schema.innodb_cmp_per_index_reset. */
|
|
static ST_FIELD_INFO i_s_cmp_per_index_fields_info[] =
|
|
{
|
|
#define IDX_DATABASE_NAME 0
|
|
{STRUCT_FLD(field_name, "database_name"),
|
|
STRUCT_FLD(field_length, 192),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_TABLE_NAME 1
|
|
{STRUCT_FLD(field_name, "table_name"),
|
|
STRUCT_FLD(field_length, 192),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_INDEX_NAME 2
|
|
{STRUCT_FLD(field_name, "index_name"),
|
|
STRUCT_FLD(field_length, 192),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_COMPRESS_OPS 3
|
|
{STRUCT_FLD(field_name, "compress_ops"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_COMPRESS_OPS_OK 4
|
|
{STRUCT_FLD(field_name, "compress_ops_ok"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_COMPRESS_TIME 5
|
|
{STRUCT_FLD(field_name, "compress_time"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_UNCOMPRESS_OPS 6
|
|
{STRUCT_FLD(field_name, "uncompress_ops"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_UNCOMPRESS_TIME 7
|
|
{STRUCT_FLD(field_name, "uncompress_time"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table
|
|
information_schema.innodb_cmp_per_index or
|
|
information_schema.innodb_cmp_per_index_reset.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmp_per_index_fill_low(
|
|
/*=======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* , /*!< in: condition (ignored) */
|
|
ibool reset) /*!< in: TRUE=reset cumulated counts */
|
|
{
|
|
TABLE* table = tables->table;
|
|
Field** fields = table->field;
|
|
int status = 0;
|
|
|
|
DBUG_ENTER("i_s_cmp_per_index_fill_low");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* Create a snapshot of the stats so we do not bump into lock
|
|
order violations with dict_sys->mutex below. */
|
|
mutex_enter(&page_zip_stat_per_index_mutex);
|
|
page_zip_stat_per_index_t snap (page_zip_stat_per_index);
|
|
mutex_exit(&page_zip_stat_per_index_mutex);
|
|
|
|
mutex_enter(&dict_sys->mutex);
|
|
|
|
page_zip_stat_per_index_t::iterator iter;
|
|
ulint i;
|
|
|
|
for (iter = snap.begin(), i = 0; iter != snap.end(); iter++, i++) {
|
|
|
|
char name[192];
|
|
dict_index_t* index = dict_index_find_on_id_low(iter->first);
|
|
|
|
if (index != NULL) {
|
|
char db_utf8[MAX_DB_UTF8_LEN];
|
|
char table_utf8[MAX_TABLE_UTF8_LEN];
|
|
|
|
dict_fs2utf8(index->table_name,
|
|
db_utf8, sizeof(db_utf8),
|
|
table_utf8, sizeof(table_utf8));
|
|
|
|
field_store_string(fields[IDX_DATABASE_NAME], db_utf8);
|
|
field_store_string(fields[IDX_TABLE_NAME], table_utf8);
|
|
field_store_index_name(fields[IDX_INDEX_NAME],
|
|
index->name);
|
|
} else {
|
|
/* index not found */
|
|
ut_snprintf(name, sizeof(name),
|
|
"index_id:" IB_ID_FMT, iter->first);
|
|
field_store_string(fields[IDX_DATABASE_NAME],
|
|
"unknown");
|
|
field_store_string(fields[IDX_TABLE_NAME],
|
|
"unknown");
|
|
field_store_string(fields[IDX_INDEX_NAME],
|
|
name);
|
|
}
|
|
|
|
fields[IDX_COMPRESS_OPS]->store(
|
|
iter->second.compressed, true);
|
|
|
|
fields[IDX_COMPRESS_OPS_OK]->store(
|
|
iter->second.compressed_ok, true);
|
|
|
|
fields[IDX_COMPRESS_TIME]->store(
|
|
iter->second.compressed_usec / 1000000, true);
|
|
|
|
fields[IDX_UNCOMPRESS_OPS]->store(
|
|
iter->second.decompressed, true);
|
|
|
|
fields[IDX_UNCOMPRESS_TIME]->store(
|
|
iter->second.decompressed_usec / 1000000, true);
|
|
|
|
if (schema_table_store_record(thd, table)) {
|
|
status = 1;
|
|
break;
|
|
}
|
|
/* Release and reacquire the dict mutex to allow other
|
|
threads to proceed. This could eventually result in the
|
|
contents of INFORMATION_SCHEMA.innodb_cmp_per_index being
|
|
inconsistent, but it is an acceptable compromise. */
|
|
if (i % 1000 == 0) {
|
|
mutex_exit(&dict_sys->mutex);
|
|
mutex_enter(&dict_sys->mutex);
|
|
}
|
|
}
|
|
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (reset) {
|
|
page_zip_reset_stat_per_index();
|
|
}
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmp_per_index.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmp_per_index_fill(
|
|
/*===================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* cond) /*!< in: condition (ignored) */
|
|
{
|
|
return(i_s_cmp_per_index_fill_low(thd, tables, cond, FALSE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmp_per_index_reset.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmp_per_index_reset_fill(
|
|
/*=========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* cond) /*!< in: condition (ignored) */
|
|
{
|
|
return(i_s_cmp_per_index_fill_low(thd, tables, cond, TRUE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table information_schema.innodb_cmp_per_index.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_cmp_per_index_init(
|
|
/*===================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_cmp_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_cmp_per_index_fields_info;
|
|
schema->fill_table = i_s_cmp_per_index_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table information_schema.innodb_cmp_per_index_reset.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_cmp_per_index_reset_init(
|
|
/*=========================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_cmp_reset_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_cmp_per_index_fields_info;
|
|
schema->fill_table = i_s_cmp_per_index_reset_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_cmp_per_index =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_CMP_PER_INDEX"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "Statistics for the InnoDB compression (per index)"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_cmp_per_index_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_cmp_per_index_reset =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_CMP_PER_INDEX_RESET"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "Statistics for the InnoDB compression (per index);"
|
|
" reset cumulated counts"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_cmp_per_index_reset_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table information_schema.innodb_cmpmem. */
|
|
static ST_FIELD_INFO i_s_cmpmem_fields_info[] =
|
|
{
|
|
{STRUCT_FLD(field_name, "page_size"),
|
|
STRUCT_FLD(field_length, 5),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Buddy Block Size"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "buffer_pool_instance"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Buffer Pool Id"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "pages_used"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Currently in Use"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "pages_free"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Currently Available"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "relocation_ops"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Total Number of Relocations"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
{STRUCT_FLD(field_name, "relocation_time"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, "Total Duration of Relocations,"
|
|
" in Seconds"),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmpmem or
|
|
innodb_cmpmem_reset.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmpmem_fill_low(
|
|
/*================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* , /*!< in: condition (ignored) */
|
|
ibool reset) /*!< in: TRUE=reset cumulated counts */
|
|
{
|
|
int status = 0;
|
|
TABLE* table = (TABLE*) tables->table;
|
|
|
|
DBUG_ENTER("i_s_cmpmem_fill_low");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
for (ulint i = 0; i < srv_buf_pool_instances; i++) {
|
|
buf_pool_t* buf_pool;
|
|
ulint zip_free_len_local[BUF_BUDDY_SIZES_MAX + 1];
|
|
buf_buddy_stat_t buddy_stat_local[BUF_BUDDY_SIZES_MAX + 1];
|
|
|
|
status = 0;
|
|
|
|
buf_pool = buf_pool_from_array(i);
|
|
|
|
/* Save buddy stats for buffer pool in local variables. */
|
|
buf_pool_mutex_enter(buf_pool);
|
|
for (uint x = 0; x <= BUF_BUDDY_SIZES; x++) {
|
|
|
|
zip_free_len_local[x] = (x < BUF_BUDDY_SIZES) ?
|
|
UT_LIST_GET_LEN(buf_pool->zip_free[x]) : 0;
|
|
|
|
buddy_stat_local[x] = buf_pool->buddy_stat[x];
|
|
|
|
if (reset) {
|
|
/* This is protected by buf_pool->mutex. */
|
|
buf_pool->buddy_stat[x].relocated = 0;
|
|
buf_pool->buddy_stat[x].relocated_usec = 0;
|
|
}
|
|
}
|
|
buf_pool_mutex_exit(buf_pool);
|
|
|
|
for (uint x = 0; x <= BUF_BUDDY_SIZES; x++) {
|
|
buf_buddy_stat_t* buddy_stat;
|
|
|
|
buddy_stat = &buddy_stat_local[x];
|
|
|
|
table->field[0]->store(BUF_BUDDY_LOW << x);
|
|
table->field[1]->store(i, true);
|
|
table->field[2]->store(buddy_stat->used, true);
|
|
table->field[3]->store(zip_free_len_local[x], true);
|
|
table->field[4]->store(buddy_stat->relocated, true);
|
|
table->field[5]->store(
|
|
buddy_stat->relocated_usec / 1000000, true);
|
|
|
|
if (schema_table_store_record(thd, table)) {
|
|
status = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (status) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmpmem.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmpmem_fill(
|
|
/*============*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* cond) /*!< in: condition (ignored) */
|
|
{
|
|
return(i_s_cmpmem_fill_low(thd, tables, cond, FALSE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_cmpmem_reset.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_cmpmem_reset_fill(
|
|
/*==================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* cond) /*!< in: condition (ignored) */
|
|
{
|
|
return(i_s_cmpmem_fill_low(thd, tables, cond, TRUE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table information_schema.innodb_cmpmem.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_cmpmem_init(
|
|
/*============*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_cmpmem_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_cmpmem_fields_info;
|
|
schema->fill_table = i_s_cmpmem_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table information_schema.innodb_cmpmem_reset.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_cmpmem_reset_init(
|
|
/*==================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_cmpmem_reset_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_cmpmem_fields_info;
|
|
schema->fill_table = i_s_cmpmem_reset_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_cmpmem =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_CMPMEM"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "Statistics for the InnoDB compressed buffer pool"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_cmpmem_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_cmpmem_reset =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_CMPMEM_RESET"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "Statistics for the InnoDB compressed buffer pool;"
|
|
" reset cumulated counts"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_cmpmem_reset_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.innodb_metrics */
|
|
static ST_FIELD_INFO innodb_metrics_fields_info[] =
|
|
{
|
|
#define METRIC_NAME 0
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_SUBSYS 1
|
|
{STRUCT_FLD(field_name, "SUBSYSTEM"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_VALUE_START 2
|
|
{STRUCT_FLD(field_name, "COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_MAX_VALUE_START 3
|
|
{STRUCT_FLD(field_name, "MAX_COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_MIN_VALUE_START 4
|
|
{STRUCT_FLD(field_name, "MIN_COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_AVG_VALUE_START 5
|
|
{STRUCT_FLD(field_name, "AVG_COUNT"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_VALUE_RESET 6
|
|
{STRUCT_FLD(field_name, "COUNT_RESET"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_MAX_VALUE_RESET 7
|
|
{STRUCT_FLD(field_name, "MAX_COUNT_RESET"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_MIN_VALUE_RESET 8
|
|
{STRUCT_FLD(field_name, "MIN_COUNT_RESET"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_AVG_VALUE_RESET 9
|
|
{STRUCT_FLD(field_name, "AVG_COUNT_RESET"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_START_TIME 10
|
|
{STRUCT_FLD(field_name, "TIME_ENABLED"),
|
|
STRUCT_FLD(field_length, 0),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_DATETIME),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_STOP_TIME 11
|
|
{STRUCT_FLD(field_name, "TIME_DISABLED"),
|
|
STRUCT_FLD(field_length, 0),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_DATETIME),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_TIME_ELAPSED 12
|
|
{STRUCT_FLD(field_name, "TIME_ELAPSED"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_RESET_TIME 13
|
|
{STRUCT_FLD(field_name, "TIME_RESET"),
|
|
STRUCT_FLD(field_length, 0),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_DATETIME),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_STATUS 14
|
|
{STRUCT_FLD(field_name, "STATUS"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_TYPE 15
|
|
{STRUCT_FLD(field_name, "TYPE"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define METRIC_DESC 16
|
|
{STRUCT_FLD(field_name, "COMMENT"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Fill the information schema metrics table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_metrics_fill(
|
|
/*=============*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
int count;
|
|
Field** fields;
|
|
double time_diff = 0;
|
|
monitor_info_t* monitor_info;
|
|
mon_type_t min_val;
|
|
mon_type_t max_val;
|
|
|
|
DBUG_ENTER("i_s_metrics_fill");
|
|
fields = table_to_fill->field;
|
|
|
|
for (count = 0; count < NUM_MONITOR; count++) {
|
|
monitor_info = srv_mon_get_info((monitor_id_t) count);
|
|
|
|
/* A good place to sanity check the Monitor ID */
|
|
ut_a(count == monitor_info->monitor_id);
|
|
|
|
/* If the item refers to a Module, nothing to fill,
|
|
continue. */
|
|
if ((monitor_info->monitor_type & MONITOR_MODULE)
|
|
|| (monitor_info->monitor_type & MONITOR_HIDDEN)) {
|
|
continue;
|
|
}
|
|
|
|
/* If this is an existing "status variable", and
|
|
its corresponding counter is still on, we need
|
|
to calculate the result from its corresponding
|
|
counter. */
|
|
if (monitor_info->monitor_type & MONITOR_EXISTING
|
|
&& MONITOR_IS_ON(count)) {
|
|
srv_mon_process_existing_counter((monitor_id_t) count,
|
|
MONITOR_GET_VALUE);
|
|
}
|
|
|
|
/* Fill in counter's basic information */
|
|
OK(field_store_string(fields[METRIC_NAME],
|
|
monitor_info->monitor_name));
|
|
|
|
OK(field_store_string(fields[METRIC_SUBSYS],
|
|
monitor_info->monitor_module));
|
|
|
|
OK(field_store_string(fields[METRIC_DESC],
|
|
monitor_info->monitor_desc));
|
|
|
|
/* Fill in counter values */
|
|
OK(fields[METRIC_VALUE_RESET]->store(
|
|
MONITOR_VALUE(count), FALSE));
|
|
|
|
OK(fields[METRIC_VALUE_START]->store(
|
|
MONITOR_VALUE_SINCE_START(count), FALSE));
|
|
|
|
/* If the max value is MAX_RESERVED, counter max
|
|
value has not been updated. Set the column value
|
|
to NULL. */
|
|
if (MONITOR_MAX_VALUE(count) == MAX_RESERVED
|
|
|| MONITOR_MAX_MIN_NOT_INIT(count)) {
|
|
fields[METRIC_MAX_VALUE_RESET]->set_null();
|
|
} else {
|
|
OK(fields[METRIC_MAX_VALUE_RESET]->store(
|
|
MONITOR_MAX_VALUE(count), FALSE));
|
|
fields[METRIC_MAX_VALUE_RESET]->set_notnull();
|
|
}
|
|
|
|
/* If the min value is MAX_RESERVED, counter min
|
|
value has not been updated. Set the column value
|
|
to NULL. */
|
|
if (MONITOR_MIN_VALUE(count) == MIN_RESERVED
|
|
|| MONITOR_MAX_MIN_NOT_INIT(count)) {
|
|
fields[METRIC_MIN_VALUE_RESET]->set_null();
|
|
} else {
|
|
OK(fields[METRIC_MIN_VALUE_RESET]->store(
|
|
MONITOR_MIN_VALUE(count), FALSE));
|
|
fields[METRIC_MIN_VALUE_RESET]->set_notnull();
|
|
}
|
|
|
|
/* Calculate the max value since counter started */
|
|
max_val = srv_mon_calc_max_since_start((monitor_id_t) count);
|
|
|
|
if (max_val == MAX_RESERVED
|
|
|| MONITOR_MAX_MIN_NOT_INIT(count)) {
|
|
fields[METRIC_MAX_VALUE_START]->set_null();
|
|
} else {
|
|
OK(fields[METRIC_MAX_VALUE_START]->store(
|
|
max_val, FALSE));
|
|
fields[METRIC_MAX_VALUE_START]->set_notnull();
|
|
}
|
|
|
|
/* Calculate the min value since counter started */
|
|
min_val = srv_mon_calc_min_since_start((monitor_id_t) count);
|
|
|
|
if (min_val == MIN_RESERVED
|
|
|| MONITOR_MAX_MIN_NOT_INIT(count)) {
|
|
fields[METRIC_MIN_VALUE_START]->set_null();
|
|
} else {
|
|
OK(fields[METRIC_MIN_VALUE_START]->store(
|
|
min_val, FALSE));
|
|
|
|
fields[METRIC_MIN_VALUE_START]->set_notnull();
|
|
}
|
|
|
|
/* If monitor has been enabled (no matter it is disabled
|
|
or not now), fill METRIC_START_TIME and METRIC_TIME_ELAPSED
|
|
field */
|
|
if (MONITOR_FIELD(count, mon_start_time)) {
|
|
OK(field_store_time_t(fields[METRIC_START_TIME],
|
|
(time_t)MONITOR_FIELD(count, mon_start_time)));
|
|
fields[METRIC_START_TIME]->set_notnull();
|
|
|
|
/* If monitor is enabled, the TIME_ELAPSED is the
|
|
time difference between current and time when monitor
|
|
is enabled. Otherwise, it is the time difference
|
|
between time when monitor is enabled and time
|
|
when it is disabled */
|
|
if (MONITOR_IS_ON(count)) {
|
|
time_diff = difftime(time(NULL),
|
|
MONITOR_FIELD(count, mon_start_time));
|
|
} else {
|
|
time_diff = difftime(
|
|
MONITOR_FIELD(count, mon_stop_time),
|
|
MONITOR_FIELD(count, mon_start_time));
|
|
}
|
|
|
|
OK(fields[METRIC_TIME_ELAPSED]->store(
|
|
time_diff));
|
|
fields[METRIC_TIME_ELAPSED]->set_notnull();
|
|
} else {
|
|
fields[METRIC_START_TIME]->set_null();
|
|
fields[METRIC_TIME_ELAPSED]->set_null();
|
|
time_diff = 0;
|
|
}
|
|
|
|
/* Unless MONITOR__NO_AVERAGE is marked, we will need
|
|
to calculate the average value. If this is a monitor set
|
|
owner marked by MONITOR_SET_OWNER, divide
|
|
the value by another counter (number of calls) designated
|
|
by monitor_info->monitor_related_id.
|
|
Otherwise average the counter value by the time between the
|
|
time that the counter is enabled and time it is disabled
|
|
or time it is sampled. */
|
|
if (!(monitor_info->monitor_type & MONITOR_NO_AVERAGE)
|
|
&& (monitor_info->monitor_type & MONITOR_SET_OWNER)
|
|
&& monitor_info->monitor_related_id) {
|
|
mon_type_t value_start
|
|
= MONITOR_VALUE_SINCE_START(
|
|
monitor_info->monitor_related_id);
|
|
|
|
if (value_start) {
|
|
OK(fields[METRIC_AVG_VALUE_START]->store(
|
|
MONITOR_VALUE_SINCE_START(count)
|
|
/ value_start, FALSE));
|
|
|
|
fields[METRIC_AVG_VALUE_START]->set_notnull();
|
|
} else {
|
|
fields[METRIC_AVG_VALUE_START]->set_null();
|
|
}
|
|
|
|
if (MONITOR_VALUE(monitor_info->monitor_related_id)) {
|
|
OK(fields[METRIC_AVG_VALUE_RESET]->store(
|
|
MONITOR_VALUE(count)
|
|
/ MONITOR_VALUE(
|
|
monitor_info->monitor_related_id),
|
|
FALSE));
|
|
} else {
|
|
fields[METRIC_AVG_VALUE_RESET]->set_null();
|
|
}
|
|
} else if (!(monitor_info->monitor_type & MONITOR_NO_AVERAGE)
|
|
&& !(monitor_info->monitor_type
|
|
& MONITOR_DISPLAY_CURRENT)) {
|
|
if (time_diff) {
|
|
OK(fields[METRIC_AVG_VALUE_START]->store(
|
|
(double) MONITOR_VALUE_SINCE_START(
|
|
count) / time_diff));
|
|
fields[METRIC_AVG_VALUE_START]->set_notnull();
|
|
} else {
|
|
fields[METRIC_AVG_VALUE_START]->set_null();
|
|
}
|
|
|
|
if (MONITOR_FIELD(count, mon_reset_time)) {
|
|
/* calculate the time difference since last
|
|
reset */
|
|
if (MONITOR_IS_ON(count)) {
|
|
time_diff = difftime(
|
|
time(NULL), MONITOR_FIELD(
|
|
count, mon_reset_time));
|
|
} else {
|
|
time_diff = difftime(
|
|
MONITOR_FIELD(count, mon_stop_time),
|
|
MONITOR_FIELD(count, mon_reset_time));
|
|
}
|
|
} else {
|
|
time_diff = 0;
|
|
}
|
|
|
|
if (time_diff) {
|
|
OK(fields[METRIC_AVG_VALUE_RESET]->store(
|
|
static_cast<double>(
|
|
MONITOR_VALUE(count) / time_diff)));
|
|
fields[METRIC_AVG_VALUE_RESET]->set_notnull();
|
|
} else {
|
|
fields[METRIC_AVG_VALUE_RESET]->set_null();
|
|
}
|
|
} else {
|
|
fields[METRIC_AVG_VALUE_START]->set_null();
|
|
fields[METRIC_AVG_VALUE_RESET]->set_null();
|
|
}
|
|
|
|
|
|
if (MONITOR_IS_ON(count)) {
|
|
/* If monitor is on, the stop time will set to NULL */
|
|
fields[METRIC_STOP_TIME]->set_null();
|
|
|
|
/* Display latest Monitor Reset Time only if Monitor
|
|
counter is on. */
|
|
if (MONITOR_FIELD(count, mon_reset_time)) {
|
|
OK(field_store_time_t(
|
|
fields[METRIC_RESET_TIME],
|
|
(time_t)MONITOR_FIELD(
|
|
count, mon_reset_time)));
|
|
fields[METRIC_RESET_TIME]->set_notnull();
|
|
} else {
|
|
fields[METRIC_RESET_TIME]->set_null();
|
|
}
|
|
|
|
/* Display the monitor status as "enabled" */
|
|
OK(field_store_string(fields[METRIC_STATUS],
|
|
"enabled"));
|
|
} else {
|
|
if (MONITOR_FIELD(count, mon_stop_time)) {
|
|
OK(field_store_time_t(fields[METRIC_STOP_TIME],
|
|
(time_t)MONITOR_FIELD(count, mon_stop_time)));
|
|
fields[METRIC_STOP_TIME]->set_notnull();
|
|
} else {
|
|
fields[METRIC_STOP_TIME]->set_null();
|
|
}
|
|
|
|
fields[METRIC_RESET_TIME]->set_null();
|
|
|
|
OK(field_store_string(fields[METRIC_STATUS],
|
|
"disabled"));
|
|
}
|
|
|
|
if (monitor_info->monitor_type & MONITOR_DISPLAY_CURRENT) {
|
|
OK(field_store_string(fields[METRIC_TYPE],
|
|
"value"));
|
|
} else if (monitor_info->monitor_type & MONITOR_EXISTING) {
|
|
OK(field_store_string(fields[METRIC_TYPE],
|
|
"status_counter"));
|
|
} else if (monitor_info->monitor_type & MONITOR_SET_OWNER) {
|
|
OK(field_store_string(fields[METRIC_TYPE],
|
|
"set_owner"));
|
|
} else if ( monitor_info->monitor_type & MONITOR_SET_MEMBER) {
|
|
OK(field_store_string(fields[METRIC_TYPE],
|
|
"set_member"));
|
|
} else {
|
|
OK(field_store_string(fields[METRIC_TYPE],
|
|
"counter"));
|
|
}
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Function to fill information schema metrics tables.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_metrics_fill_table(
|
|
/*===================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
DBUG_ENTER("i_s_metrics_fill_table");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
i_s_metrics_fill(thd, tables->table);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_metrics
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_metrics_init(
|
|
/*================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_metrics_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_metrics_fields_info;
|
|
schema->fill_table = i_s_metrics_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_metrics =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_METRICS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB Metrics Info"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_metrics_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.innodb_ft_default_stopword */
|
|
static ST_FIELD_INFO i_s_stopword_fields_info[] =
|
|
{
|
|
#define STOPWORD_VALUE 0
|
|
{STRUCT_FLD(field_name, "value"),
|
|
STRUCT_FLD(field_length, TRX_ID_MAX_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table information_schema.innodb_ft_default_stopword.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_stopword_fill(
|
|
/*==============*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
Field** fields;
|
|
ulint i = 0;
|
|
TABLE* table = (TABLE*) tables->table;
|
|
|
|
DBUG_ENTER("i_s_stopword_fill");
|
|
|
|
fields = table->field;
|
|
|
|
/* Fill with server default stopword list in array
|
|
fts_default_stopword */
|
|
while (fts_default_stopword[i]) {
|
|
OK(field_store_string(fields[STOPWORD_VALUE],
|
|
fts_default_stopword[i]));
|
|
|
|
OK(schema_table_store_record(thd, table));
|
|
i++;
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table information_schema.innodb_ft_default_stopword.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_stopword_init(
|
|
/*==============*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_stopword_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_stopword_fields_info;
|
|
schema->fill_table = i_s_stopword_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_ft_default_stopword =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_FT_DEFAULT_STOPWORD"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "Default stopword list for InnoDB Full Text Search"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_stopword_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_FT_DELETED
|
|
INFORMATION_SCHEMA.INNODB_FT_BEING_DELETED */
|
|
static ST_FIELD_INFO i_s_fts_doc_fields_info[] =
|
|
{
|
|
#define I_S_FTS_DOC_ID 0
|
|
{STRUCT_FLD(field_name, "DOC_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table INFORMATION_SCHEMA.INNODB_FT_DELETED or
|
|
INFORMATION_SCHEMA.INNODB_FT_BEING_DELETED
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_deleted_generic_fill(
|
|
/*=========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
ibool being_deleted) /*!< in: BEING_DELTED table */
|
|
{
|
|
Field** fields;
|
|
TABLE* table = (TABLE*) tables->table;
|
|
trx_t* trx;
|
|
fts_table_t fts_table;
|
|
fts_doc_ids_t* deleted;
|
|
dict_table_t* user_table;
|
|
|
|
DBUG_ENTER("i_s_fts_deleted_generic_fill");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
if (!fts_internal_tbl_name) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* Prevent DDL to drop fts aux tables. */
|
|
rw_lock_s_lock(dict_operation_lock);
|
|
|
|
user_table = dict_table_open_on_name(
|
|
fts_internal_tbl_name, FALSE, FALSE, DICT_ERR_IGNORE_NONE);
|
|
|
|
if (!user_table) {
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
} else if (!dict_table_has_fts_index(user_table)) {
|
|
dict_table_close(user_table, FALSE, FALSE);
|
|
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
deleted = fts_doc_ids_create();
|
|
|
|
trx = trx_allocate_for_background();
|
|
trx->op_info = "Select for FTS DELETE TABLE";
|
|
|
|
FTS_INIT_FTS_TABLE(&fts_table,
|
|
(being_deleted) ? "BEING_DELETED" : "DELETED",
|
|
FTS_COMMON_TABLE, user_table);
|
|
|
|
fts_table_fetch_doc_ids(trx, &fts_table, deleted);
|
|
|
|
fields = table->field;
|
|
|
|
for (ulint j = 0; j < ib_vector_size(deleted->doc_ids); ++j) {
|
|
doc_id_t doc_id;
|
|
|
|
doc_id = *(doc_id_t*) ib_vector_get_const(deleted->doc_ids, j);
|
|
|
|
OK(fields[I_S_FTS_DOC_ID]->store(doc_id, true));
|
|
|
|
OK(schema_table_store_record(thd, table));
|
|
}
|
|
|
|
trx_free_for_background(trx);
|
|
|
|
fts_doc_ids_free(deleted);
|
|
|
|
dict_table_close(user_table, FALSE, FALSE);
|
|
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table INFORMATION_SCHEMA.INNODB_FT_DELETED
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_deleted_fill(
|
|
/*=================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
DBUG_ENTER("i_s_fts_deleted_fill");
|
|
|
|
DBUG_RETURN(i_s_fts_deleted_generic_fill(thd, tables, FALSE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_FT_DELETED
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_fts_deleted_init(
|
|
/*=================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_fts_deleted_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_fts_doc_fields_info;
|
|
schema->fill_table = i_s_fts_deleted_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_ft_deleted =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_FT_DELETED"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "INNODB AUXILIARY FTS DELETED TABLE"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_fts_deleted_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table INFORMATION_SCHEMA.INNODB_FT_BEING_DELETED
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_being_deleted_fill(
|
|
/*=======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
DBUG_ENTER("i_s_fts_being_deleted_fill");
|
|
|
|
DBUG_RETURN(i_s_fts_deleted_generic_fill(thd, tables, TRUE));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_FT_BEING_DELETED
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_fts_being_deleted_init(
|
|
/*=======================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_fts_deleted_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_fts_doc_fields_info;
|
|
schema->fill_table = i_s_fts_being_deleted_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_ft_being_deleted =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_FT_BEING_DELETED"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "INNODB AUXILIARY FTS BEING DELETED TABLE"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_fts_being_deleted_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_FT_INDEX_CACHED and
|
|
INFORMATION_SCHEMA.INNODB_FT_INDEX_TABLE */
|
|
static ST_FIELD_INFO i_s_fts_index_fields_info[] =
|
|
{
|
|
#define I_S_FTS_WORD 0
|
|
{STRUCT_FLD(field_name, "WORD"),
|
|
STRUCT_FLD(field_length, FTS_MAX_WORD_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define I_S_FTS_FIRST_DOC_ID 1
|
|
{STRUCT_FLD(field_name, "FIRST_DOC_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define I_S_FTS_LAST_DOC_ID 2
|
|
{STRUCT_FLD(field_name, "LAST_DOC_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define I_S_FTS_DOC_COUNT 3
|
|
{STRUCT_FLD(field_name, "DOC_COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define I_S_FTS_ILIST_DOC_ID 4
|
|
{STRUCT_FLD(field_name, "DOC_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define I_S_FTS_ILIST_DOC_POS 5
|
|
{STRUCT_FLD(field_name, "POSITION"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Go through the Doc Node and its ilist, fill the dynamic table
|
|
INFORMATION_SCHEMA.INNODB_FT_INDEX_CACHED for one FTS index on the table.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_index_cache_fill_one_index(
|
|
/*===============================*/
|
|
fts_index_cache_t* index_cache, /*!< in: FTS index cache */
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables) /*!< in/out: tables to fill */
|
|
{
|
|
TABLE* table = (TABLE*) tables->table;
|
|
Field** fields;
|
|
CHARSET_INFO* index_charset;
|
|
const ib_rbt_node_t* rbt_node;
|
|
fts_string_t conv_str;
|
|
uint dummy_errors;
|
|
char* word_str;
|
|
|
|
DBUG_ENTER("i_s_fts_index_cache_fill_one_index");
|
|
|
|
fields = table->field;
|
|
|
|
index_charset = index_cache->charset;
|
|
conv_str.f_len = system_charset_info->mbmaxlen
|
|
* FTS_MAX_WORD_LEN_IN_CHAR;
|
|
conv_str.f_str = static_cast<byte*>(ut_malloc_nokey(conv_str.f_len));
|
|
conv_str.f_n_char = 0;
|
|
|
|
/* Go through each word in the index cache */
|
|
for (rbt_node = rbt_first(index_cache->words);
|
|
rbt_node;
|
|
rbt_node = rbt_next(index_cache->words, rbt_node)) {
|
|
fts_tokenizer_word_t* word;
|
|
|
|
word = rbt_value(fts_tokenizer_word_t, rbt_node);
|
|
|
|
/* Convert word from index charset to system_charset_info */
|
|
if (index_charset->cset != system_charset_info->cset) {
|
|
conv_str.f_n_char = my_convert(
|
|
reinterpret_cast<char*>(conv_str.f_str),
|
|
static_cast<uint32>(conv_str.f_len),
|
|
system_charset_info,
|
|
reinterpret_cast<char*>(word->text.f_str),
|
|
static_cast<uint32>(word->text.f_len),
|
|
index_charset, &dummy_errors);
|
|
ut_ad(conv_str.f_n_char <= conv_str.f_len);
|
|
conv_str.f_str[conv_str.f_n_char] = 0;
|
|
word_str = reinterpret_cast<char*>(conv_str.f_str);
|
|
} else {
|
|
word_str = reinterpret_cast<char*>(word->text.f_str);
|
|
}
|
|
|
|
/* Decrypt the ilist, and display Dod ID and word position */
|
|
for (ulint i = 0; i < ib_vector_size(word->nodes); i++) {
|
|
fts_node_t* node;
|
|
byte* ptr;
|
|
ulint decoded = 0;
|
|
doc_id_t doc_id = 0;
|
|
|
|
node = static_cast<fts_node_t*> (ib_vector_get(
|
|
word->nodes, i));
|
|
|
|
ptr = node->ilist;
|
|
|
|
while (decoded < node->ilist_size) {
|
|
ulint pos = fts_decode_vlc(&ptr);
|
|
|
|
doc_id += pos;
|
|
|
|
/* Get position info */
|
|
while (*ptr) {
|
|
pos = fts_decode_vlc(&ptr);
|
|
|
|
OK(field_store_string(
|
|
fields[I_S_FTS_WORD],
|
|
word_str));
|
|
|
|
OK(fields[I_S_FTS_FIRST_DOC_ID]->store(
|
|
node->first_doc_id,
|
|
true));
|
|
|
|
OK(fields[I_S_FTS_LAST_DOC_ID]->store(
|
|
node->last_doc_id,
|
|
true));
|
|
|
|
OK(fields[I_S_FTS_DOC_COUNT]->store(
|
|
node->doc_count, true));
|
|
|
|
OK(fields[I_S_FTS_ILIST_DOC_ID]->store(
|
|
doc_id, true));
|
|
|
|
OK(fields[I_S_FTS_ILIST_DOC_POS]->store(
|
|
pos, true));
|
|
|
|
OK(schema_table_store_record(
|
|
thd, table));
|
|
}
|
|
|
|
++ptr;
|
|
|
|
decoded = ptr - (byte*) node->ilist;
|
|
}
|
|
}
|
|
}
|
|
|
|
ut_free(conv_str.f_str);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Fill the dynamic table INFORMATION_SCHEMA.INNODB_FT_INDEX_CACHED
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_index_cache_fill(
|
|
/*=====================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
dict_table_t* user_table;
|
|
fts_cache_t* cache;
|
|
|
|
DBUG_ENTER("i_s_fts_index_cache_fill");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
if (!fts_internal_tbl_name) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
user_table = dict_table_open_on_name(
|
|
fts_internal_tbl_name, FALSE, FALSE, DICT_ERR_IGNORE_NONE);
|
|
|
|
if (!user_table) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
if (user_table->fts == NULL || user_table->fts->cache == NULL) {
|
|
dict_table_close(user_table, FALSE, FALSE);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
cache = user_table->fts->cache;
|
|
|
|
ut_a(cache);
|
|
|
|
for (ulint i = 0; i < ib_vector_size(cache->indexes); i++) {
|
|
fts_index_cache_t* index_cache;
|
|
|
|
index_cache = static_cast<fts_index_cache_t*> (
|
|
ib_vector_get(cache->indexes, i));
|
|
|
|
i_s_fts_index_cache_fill_one_index(index_cache, thd, tables);
|
|
}
|
|
|
|
dict_table_close(user_table, FALSE, FALSE);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_FT_INDEX_CACHE
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_fts_index_cache_init(
|
|
/*=====================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_fts_index_cache_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_fts_index_fields_info;
|
|
schema->fill_table = i_s_fts_index_cache_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_ft_index_cache =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_FT_INDEX_CACHE"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "INNODB AUXILIARY FTS INDEX CACHED"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_fts_index_cache_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Go through a FTS index auxiliary table, fetch its rows and fill
|
|
FTS word cache structure.
|
|
@return DB_SUCCESS on success, otherwise error code */
|
|
static
|
|
dberr_t
|
|
i_s_fts_index_table_fill_selected(
|
|
/*==============================*/
|
|
dict_index_t* index, /*!< in: FTS index */
|
|
ib_vector_t* words, /*!< in/out: vector to hold
|
|
fetched words */
|
|
ulint selected, /*!< in: selected FTS index */
|
|
fts_string_t* word) /*!< in: word to select */
|
|
{
|
|
pars_info_t* info;
|
|
fts_table_t fts_table;
|
|
trx_t* trx;
|
|
que_t* graph;
|
|
dberr_t error;
|
|
fts_fetch_t fetch;
|
|
char table_name[MAX_FULL_NAME_LEN];
|
|
|
|
info = pars_info_create();
|
|
|
|
fetch.read_arg = words;
|
|
fetch.read_record = fts_optimize_index_fetch_node;
|
|
fetch.total_memory = 0;
|
|
|
|
DBUG_EXECUTE_IF("fts_instrument_result_cache_limit",
|
|
fts_result_cache_limit = 8192;
|
|
);
|
|
|
|
trx = trx_allocate_for_background();
|
|
|
|
trx->op_info = "fetching FTS index nodes";
|
|
|
|
pars_info_bind_function(info, "my_func", fetch.read_record, &fetch);
|
|
pars_info_bind_varchar_literal(info, "word", word->f_str, word->f_len);
|
|
|
|
FTS_INIT_INDEX_TABLE(&fts_table, fts_get_suffix(selected),
|
|
FTS_INDEX_TABLE, index);
|
|
fts_get_table_name(&fts_table, table_name);
|
|
pars_info_bind_id(info, true, "table_name", table_name);
|
|
|
|
graph = fts_parse_sql(
|
|
&fts_table, info,
|
|
"DECLARE FUNCTION my_func;\n"
|
|
"DECLARE CURSOR c IS"
|
|
" SELECT word, doc_count, first_doc_id, last_doc_id,"
|
|
" ilist\n"
|
|
" FROM $table_name WHERE word >= :word;\n"
|
|
"BEGIN\n"
|
|
"\n"
|
|
"OPEN c;\n"
|
|
"WHILE 1 = 1 LOOP\n"
|
|
" FETCH c INTO my_func();\n"
|
|
" IF c % NOTFOUND THEN\n"
|
|
" EXIT;\n"
|
|
" END IF;\n"
|
|
"END LOOP;\n"
|
|
"CLOSE c;");
|
|
|
|
for (;;) {
|
|
error = fts_eval_sql(trx, graph);
|
|
|
|
if (error == DB_SUCCESS) {
|
|
fts_sql_commit(trx);
|
|
|
|
break;
|
|
} else {
|
|
fts_sql_rollback(trx);
|
|
|
|
if (error == DB_LOCK_WAIT_TIMEOUT) {
|
|
ib::warn() << "Lock wait timeout reading"
|
|
" FTS index. Retrying!";
|
|
|
|
trx->error_state = DB_SUCCESS;
|
|
} else {
|
|
ib::error() << "Error occurred while reading"
|
|
" FTS index: " << ut_strerr(error);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
mutex_enter(&dict_sys->mutex);
|
|
que_graph_free(graph);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
trx_free_for_background(trx);
|
|
|
|
if (fetch.total_memory >= fts_result_cache_limit) {
|
|
error = DB_FTS_EXCEED_RESULT_CACHE_LIMIT;
|
|
}
|
|
|
|
return(error);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Free words. */
|
|
static
|
|
void
|
|
i_s_fts_index_table_free_one_fetch(
|
|
/*===============================*/
|
|
ib_vector_t* words) /*!< in: words fetched */
|
|
{
|
|
for (ulint i = 0; i < ib_vector_size(words); i++) {
|
|
fts_word_t* word;
|
|
|
|
word = static_cast<fts_word_t*>(ib_vector_get(words, i));
|
|
|
|
for (ulint j = 0; j < ib_vector_size(word->nodes); j++) {
|
|
fts_node_t* node;
|
|
|
|
node = static_cast<fts_node_t*> (ib_vector_get(
|
|
word->nodes, j));
|
|
ut_free(node->ilist);
|
|
}
|
|
|
|
fts_word_free(word);
|
|
}
|
|
|
|
ib_vector_reset(words);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Go through words, fill INFORMATION_SCHEMA.INNODB_FT_INDEX_TABLE.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_index_table_fill_one_fetch(
|
|
/*===============================*/
|
|
CHARSET_INFO* index_charset, /*!< in: FTS index charset */
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
ib_vector_t* words, /*!< in: words fetched */
|
|
fts_string_t* conv_str, /*!< in: string for conversion*/
|
|
bool has_more) /*!< in: has more to fetch */
|
|
{
|
|
TABLE* table = (TABLE*) tables->table;
|
|
Field** fields;
|
|
uint dummy_errors;
|
|
char* word_str;
|
|
ulint words_size;
|
|
int ret = 0;
|
|
|
|
DBUG_ENTER("i_s_fts_index_table_fill_one_fetch");
|
|
|
|
fields = table->field;
|
|
|
|
words_size = ib_vector_size(words);
|
|
if (has_more) {
|
|
/* the last word is not fetched completely. */
|
|
ut_ad(words_size > 1);
|
|
words_size -= 1;
|
|
}
|
|
|
|
/* Go through each word in the index cache */
|
|
for (ulint i = 0; i < words_size; i++) {
|
|
fts_word_t* word;
|
|
|
|
word = static_cast<fts_word_t*>(ib_vector_get(words, i));
|
|
|
|
word->text.f_str[word->text.f_len] = 0;
|
|
|
|
/* Convert word from index charset to system_charset_info */
|
|
if (index_charset->cset != system_charset_info->cset) {
|
|
conv_str->f_n_char = my_convert(
|
|
reinterpret_cast<char*>(conv_str->f_str),
|
|
static_cast<uint32>(conv_str->f_len),
|
|
system_charset_info,
|
|
reinterpret_cast<char*>(word->text.f_str),
|
|
static_cast<uint32>(word->text.f_len),
|
|
index_charset, &dummy_errors);
|
|
ut_ad(conv_str->f_n_char <= conv_str->f_len);
|
|
conv_str->f_str[conv_str->f_n_char] = 0;
|
|
word_str = reinterpret_cast<char*>(conv_str->f_str);
|
|
} else {
|
|
word_str = reinterpret_cast<char*>(word->text.f_str);
|
|
}
|
|
|
|
/* Decrypt the ilist, and display Dod ID and word position */
|
|
for (ulint i = 0; i < ib_vector_size(word->nodes); i++) {
|
|
fts_node_t* node;
|
|
byte* ptr;
|
|
ulint decoded = 0;
|
|
doc_id_t doc_id = 0;
|
|
|
|
node = static_cast<fts_node_t*> (ib_vector_get(
|
|
word->nodes, i));
|
|
|
|
ptr = node->ilist;
|
|
|
|
while (decoded < node->ilist_size) {
|
|
ulint pos = fts_decode_vlc(&ptr);
|
|
|
|
doc_id += pos;
|
|
|
|
/* Get position info */
|
|
while (*ptr) {
|
|
pos = fts_decode_vlc(&ptr);
|
|
|
|
OK(field_store_string(
|
|
fields[I_S_FTS_WORD],
|
|
word_str));
|
|
|
|
OK(fields[I_S_FTS_FIRST_DOC_ID]->store(
|
|
longlong(node->first_doc_id), true));
|
|
|
|
OK(fields[I_S_FTS_LAST_DOC_ID]->store(
|
|
longlong(node->last_doc_id), true));
|
|
|
|
OK(fields[I_S_FTS_DOC_COUNT]->store(
|
|
node->doc_count, true));
|
|
|
|
OK(fields[I_S_FTS_ILIST_DOC_ID]->store(
|
|
longlong(doc_id), true));
|
|
|
|
OK(fields[I_S_FTS_ILIST_DOC_POS]->store(
|
|
pos, true));
|
|
|
|
OK(schema_table_store_record(
|
|
thd, table));
|
|
}
|
|
|
|
++ptr;
|
|
|
|
decoded = ptr - (byte*) node->ilist;
|
|
}
|
|
}
|
|
}
|
|
|
|
i_s_fts_index_table_free_one_fetch(words);
|
|
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Go through a FTS index and its auxiliary tables, fetch rows in each table
|
|
and fill INFORMATION_SCHEMA.INNODB_FT_INDEX_TABLE.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_index_table_fill_one_index(
|
|
/*===============================*/
|
|
dict_index_t* index, /*!< in: FTS index */
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables) /*!< in/out: tables to fill */
|
|
{
|
|
ib_vector_t* words;
|
|
mem_heap_t* heap;
|
|
CHARSET_INFO* index_charset;
|
|
fts_string_t conv_str;
|
|
dberr_t error;
|
|
int ret = 0;
|
|
|
|
DBUG_ENTER("i_s_fts_index_table_fill_one_index");
|
|
DBUG_ASSERT(!dict_index_is_online_ddl(index));
|
|
|
|
heap = mem_heap_create(1024);
|
|
|
|
words = ib_vector_create(ib_heap_allocator_create(heap),
|
|
sizeof(fts_word_t), 256);
|
|
|
|
index_charset = fts_index_get_charset(index);
|
|
conv_str.f_len = system_charset_info->mbmaxlen
|
|
* FTS_MAX_WORD_LEN_IN_CHAR;
|
|
conv_str.f_str = static_cast<byte*>(ut_malloc_nokey(conv_str.f_len));
|
|
conv_str.f_n_char = 0;
|
|
|
|
/* Iterate through each auxiliary table as described in
|
|
fts_index_selector */
|
|
for (ulint selected = 0; selected < FTS_NUM_AUX_INDEX; selected++) {
|
|
fts_string_t word;
|
|
bool has_more = false;
|
|
|
|
word.f_str = NULL;
|
|
word.f_len = 0;
|
|
word.f_n_char = 0;
|
|
|
|
do {
|
|
/* Fetch from index */
|
|
error = i_s_fts_index_table_fill_selected(
|
|
index, words, selected, &word);
|
|
|
|
if (error == DB_SUCCESS) {
|
|
has_more = false;
|
|
} else if (error == DB_FTS_EXCEED_RESULT_CACHE_LIMIT) {
|
|
has_more = true;
|
|
} else {
|
|
i_s_fts_index_table_free_one_fetch(words);
|
|
ret = 1;
|
|
goto func_exit;
|
|
}
|
|
|
|
if (has_more) {
|
|
fts_word_t* last_word;
|
|
|
|
/* Prepare start point for next fetch */
|
|
last_word = static_cast<fts_word_t*>(ib_vector_last(words));
|
|
ut_ad(last_word != NULL);
|
|
fts_string_dup(&word, &last_word->text, heap);
|
|
}
|
|
|
|
/* Fill into tables */
|
|
ret = i_s_fts_index_table_fill_one_fetch(
|
|
index_charset, thd, tables, words, &conv_str, has_more);
|
|
|
|
if (ret != 0) {
|
|
i_s_fts_index_table_free_one_fetch(words);
|
|
goto func_exit;
|
|
}
|
|
} while (has_more);
|
|
}
|
|
|
|
func_exit:
|
|
ut_free(conv_str.f_str);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(ret);
|
|
}
|
|
/*******************************************************************//**
|
|
Fill the dynamic table INFORMATION_SCHEMA.INNODB_FT_INDEX_TABLE
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_index_table_fill(
|
|
/*=====================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
dict_table_t* user_table;
|
|
dict_index_t* index;
|
|
|
|
DBUG_ENTER("i_s_fts_index_table_fill");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
if (!fts_internal_tbl_name) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* Prevent DDL to drop fts aux tables. */
|
|
rw_lock_s_lock(dict_operation_lock);
|
|
|
|
user_table = dict_table_open_on_name(
|
|
fts_internal_tbl_name, FALSE, FALSE, DICT_ERR_IGNORE_NONE);
|
|
|
|
if (!user_table) {
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
for (index = dict_table_get_first_index(user_table);
|
|
index; index = dict_table_get_next_index(index)) {
|
|
if (index->type & DICT_FTS) {
|
|
i_s_fts_index_table_fill_one_index(index, thd, tables);
|
|
}
|
|
}
|
|
|
|
dict_table_close(user_table, FALSE, FALSE);
|
|
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_FT_INDEX_TABLE
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_fts_index_table_init(
|
|
/*=====================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_fts_index_table_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_fts_index_fields_info;
|
|
schema->fill_table = i_s_fts_index_table_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_ft_index_table =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_FT_INDEX_TABLE"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "INNODB AUXILIARY FTS INDEX TABLE"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_fts_index_table_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_FT_CONFIG */
|
|
static ST_FIELD_INFO i_s_fts_config_fields_info[] =
|
|
{
|
|
#define FTS_CONFIG_KEY 0
|
|
{STRUCT_FLD(field_name, "KEY"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define FTS_CONFIG_VALUE 1
|
|
{STRUCT_FLD(field_name, "VALUE"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
static const char* fts_config_key[] = {
|
|
FTS_OPTIMIZE_LIMIT_IN_SECS,
|
|
FTS_SYNCED_DOC_ID,
|
|
FTS_STOPWORD_TABLE_NAME,
|
|
FTS_USE_STOPWORD,
|
|
NULL
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill the dynamic table INFORMATION_SCHEMA.INNODB_FT_CONFIG
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_fts_config_fill(
|
|
/*================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
Field** fields;
|
|
TABLE* table = (TABLE*) tables->table;
|
|
trx_t* trx;
|
|
fts_table_t fts_table;
|
|
dict_table_t* user_table;
|
|
ulint i = 0;
|
|
dict_index_t* index = NULL;
|
|
unsigned char str[FTS_MAX_CONFIG_VALUE_LEN + 1];
|
|
|
|
DBUG_ENTER("i_s_fts_config_fill");
|
|
|
|
/* deny access to non-superusers */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
if (!fts_internal_tbl_name) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
fields = table->field;
|
|
|
|
/* Prevent DDL to drop fts aux tables. */
|
|
rw_lock_s_lock(dict_operation_lock);
|
|
|
|
user_table = dict_table_open_on_name(
|
|
fts_internal_tbl_name, FALSE, FALSE, DICT_ERR_IGNORE_NONE);
|
|
|
|
if (!user_table) {
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
} else if (!dict_table_has_fts_index(user_table)) {
|
|
dict_table_close(user_table, FALSE, FALSE);
|
|
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
trx = trx_allocate_for_background();
|
|
trx->op_info = "Select for FTS CONFIG TABLE";
|
|
|
|
FTS_INIT_FTS_TABLE(&fts_table, "CONFIG", FTS_COMMON_TABLE, user_table);
|
|
|
|
if (!ib_vector_is_empty(user_table->fts->indexes)) {
|
|
index = (dict_index_t*) ib_vector_getp_const(
|
|
user_table->fts->indexes, 0);
|
|
DBUG_ASSERT(!dict_index_is_online_ddl(index));
|
|
}
|
|
|
|
while (fts_config_key[i]) {
|
|
fts_string_t value;
|
|
char* key_name;
|
|
ulint allocated = FALSE;
|
|
|
|
value.f_len = FTS_MAX_CONFIG_VALUE_LEN;
|
|
|
|
value.f_str = str;
|
|
|
|
if (index
|
|
&& strcmp(fts_config_key[i], FTS_TOTAL_WORD_COUNT) == 0) {
|
|
key_name = fts_config_create_index_param_name(
|
|
fts_config_key[i], index);
|
|
allocated = TRUE;
|
|
} else {
|
|
key_name = (char*) fts_config_key[i];
|
|
}
|
|
|
|
fts_config_get_value(trx, &fts_table, key_name, &value);
|
|
|
|
if (allocated) {
|
|
ut_free(key_name);
|
|
}
|
|
|
|
OK(field_store_string(
|
|
fields[FTS_CONFIG_KEY], fts_config_key[i]));
|
|
|
|
OK(field_store_string(
|
|
fields[FTS_CONFIG_VALUE], (const char*) value.f_str));
|
|
|
|
OK(schema_table_store_record(thd, table));
|
|
|
|
i++;
|
|
}
|
|
|
|
fts_sql_commit(trx);
|
|
|
|
trx_free_for_background(trx);
|
|
|
|
dict_table_close(user_table, FALSE, FALSE);
|
|
|
|
rw_lock_s_unlock(dict_operation_lock);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_FT_CONFIG
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_fts_config_init(
|
|
/*=================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_fts_config_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = i_s_fts_config_fields_info;
|
|
schema->fill_table = i_s_fts_config_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_ft_config =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_FT_CONFIG"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "INNODB AUXILIARY FTS CONFIG TABLE"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_fts_config_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INNODB_BUFFER_POOL_STATS. */
|
|
static ST_FIELD_INFO i_s_innodb_buffer_stats_fields_info[] =
|
|
{
|
|
#define IDX_BUF_STATS_POOL_ID 0
|
|
{STRUCT_FLD(field_name, "POOL_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_POOL_SIZE 1
|
|
{STRUCT_FLD(field_name, "POOL_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_FREE_BUFFERS 2
|
|
{STRUCT_FLD(field_name, "FREE_BUFFERS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_LRU_LEN 3
|
|
{STRUCT_FLD(field_name, "DATABASE_PAGES"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_OLD_LRU_LEN 4
|
|
{STRUCT_FLD(field_name, "OLD_DATABASE_PAGES"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_FLUSH_LIST_LEN 5
|
|
{STRUCT_FLD(field_name, "MODIFIED_DATABASE_PAGES"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PENDING_ZIP 6
|
|
{STRUCT_FLD(field_name, "PENDING_DECOMPRESS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PENDING_READ 7
|
|
{STRUCT_FLD(field_name, "PENDING_READS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_FLUSH_LRU 8
|
|
{STRUCT_FLD(field_name, "PENDING_FLUSH_LRU"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_FLUSH_LIST 9
|
|
{STRUCT_FLD(field_name, "PENDING_FLUSH_LIST"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_YOUNG 10
|
|
{STRUCT_FLD(field_name, "PAGES_MADE_YOUNG"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_NOT_YOUNG 11
|
|
{STRUCT_FLD(field_name, "PAGES_NOT_MADE_YOUNG"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_YOUNG_RATE 12
|
|
{STRUCT_FLD(field_name, "PAGES_MADE_YOUNG_RATE"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_NOT_YOUNG_RATE 13
|
|
{STRUCT_FLD(field_name, "PAGES_MADE_NOT_YOUNG_RATE"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_READ 14
|
|
{STRUCT_FLD(field_name, "NUMBER_PAGES_READ"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_CREATED 15
|
|
{STRUCT_FLD(field_name, "NUMBER_PAGES_CREATED"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_WRITTEN 16
|
|
{STRUCT_FLD(field_name, "NUMBER_PAGES_WRITTEN"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_READ_RATE 17
|
|
{STRUCT_FLD(field_name, "PAGES_READ_RATE"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_CREATE_RATE 18
|
|
{STRUCT_FLD(field_name, "PAGES_CREATE_RATE"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_PAGE_WRITTEN_RATE 19
|
|
{STRUCT_FLD(field_name, "PAGES_WRITTEN_RATE"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_GET 20
|
|
{STRUCT_FLD(field_name, "NUMBER_PAGES_GET"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_HIT_RATE 21
|
|
{STRUCT_FLD(field_name, "HIT_RATE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_MADE_YOUNG_PCT 22
|
|
{STRUCT_FLD(field_name, "YOUNG_MAKE_PER_THOUSAND_GETS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_NOT_MADE_YOUNG_PCT 23
|
|
{STRUCT_FLD(field_name, "NOT_YOUNG_MAKE_PER_THOUSAND_GETS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_READ_AHREAD 24
|
|
{STRUCT_FLD(field_name, "NUMBER_PAGES_READ_AHEAD"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_READ_AHEAD_EVICTED 25
|
|
{STRUCT_FLD(field_name, "NUMBER_READ_AHEAD_EVICTED"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_READ_AHEAD_RATE 26
|
|
{STRUCT_FLD(field_name, "READ_AHEAD_RATE"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_READ_AHEAD_EVICT_RATE 27
|
|
{STRUCT_FLD(field_name, "READ_AHEAD_EVICTED_RATE"),
|
|
STRUCT_FLD(field_length, MAX_FLOAT_STR_LENGTH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_FLOAT),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_LRU_IO_SUM 28
|
|
{STRUCT_FLD(field_name, "LRU_IO_TOTAL"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_LRU_IO_CUR 29
|
|
{STRUCT_FLD(field_name, "LRU_IO_CURRENT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_UNZIP_SUM 30
|
|
{STRUCT_FLD(field_name, "UNCOMPRESS_TOTAL"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_STATS_UNZIP_CUR 31
|
|
{STRUCT_FLD(field_name, "UNCOMPRESS_CURRENT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill Information Schema table INNODB_BUFFER_POOL_STATS for a particular
|
|
buffer pool
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_stats_fill(
|
|
/*==================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
const buf_pool_info_t* info) /*!< in: buffer pool
|
|
information */
|
|
{
|
|
TABLE* table;
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_innodb_stats_fill");
|
|
|
|
table = tables->table;
|
|
|
|
fields = table->field;
|
|
|
|
OK(fields[IDX_BUF_STATS_POOL_ID]->store(
|
|
info->pool_unique_id, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_POOL_SIZE]->store(
|
|
info->pool_size, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_LRU_LEN]->store(
|
|
info->lru_len, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_OLD_LRU_LEN]->store(
|
|
info->old_lru_len, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_FREE_BUFFERS]->store(
|
|
info->free_list_len, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_FLUSH_LIST_LEN]->store(
|
|
info->flush_list_len, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PENDING_ZIP]->store(
|
|
info->n_pend_unzip, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PENDING_READ]->store(
|
|
info->n_pend_reads, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_FLUSH_LRU]->store(
|
|
info->n_pending_flush_lru, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_FLUSH_LIST]->store(
|
|
info->n_pending_flush_list, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_YOUNG]->store(
|
|
info->n_pages_made_young, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_NOT_YOUNG]->store(
|
|
info->n_pages_not_made_young, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_YOUNG_RATE]->store(
|
|
info->page_made_young_rate));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_NOT_YOUNG_RATE]->store(
|
|
info->page_not_made_young_rate));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_READ]->store(
|
|
info->n_pages_read, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_CREATED]->store(
|
|
info->n_pages_created, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_WRITTEN]->store(
|
|
info->n_pages_written, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_GET]->store(
|
|
info->n_page_gets, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_READ_RATE]->store(
|
|
info->pages_read_rate));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_CREATE_RATE]->store(
|
|
info->pages_created_rate));
|
|
|
|
OK(fields[IDX_BUF_STATS_PAGE_WRITTEN_RATE]->store(
|
|
info->pages_written_rate));
|
|
|
|
if (info->n_page_get_delta) {
|
|
if (info->page_read_delta <= info->n_page_get_delta) {
|
|
OK(fields[IDX_BUF_STATS_HIT_RATE]->store(
|
|
static_cast<double>(
|
|
1000 - (1000 * info->page_read_delta
|
|
/ info->n_page_get_delta))));
|
|
} else {
|
|
OK(fields[IDX_BUF_STATS_HIT_RATE]->store(0));
|
|
}
|
|
|
|
OK(fields[IDX_BUF_STATS_MADE_YOUNG_PCT]->store(
|
|
1000 * info->young_making_delta
|
|
/ info->n_page_get_delta, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_NOT_MADE_YOUNG_PCT]->store(
|
|
1000 * info->not_young_making_delta
|
|
/ info->n_page_get_delta, true));
|
|
} else {
|
|
OK(fields[IDX_BUF_STATS_HIT_RATE]->store(0, true));
|
|
OK(fields[IDX_BUF_STATS_MADE_YOUNG_PCT]->store(0, true));
|
|
OK(fields[IDX_BUF_STATS_NOT_MADE_YOUNG_PCT]->store(0, true));
|
|
}
|
|
|
|
OK(fields[IDX_BUF_STATS_READ_AHREAD]->store(
|
|
info->n_ra_pages_read, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_READ_AHEAD_EVICTED]->store(
|
|
info->n_ra_pages_evicted, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_READ_AHEAD_RATE]->store(
|
|
info->pages_readahead_rate));
|
|
|
|
OK(fields[IDX_BUF_STATS_READ_AHEAD_EVICT_RATE]->store(
|
|
info->pages_evicted_rate));
|
|
|
|
OK(fields[IDX_BUF_STATS_LRU_IO_SUM]->store(
|
|
info->io_sum, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_LRU_IO_CUR]->store(
|
|
info->io_cur, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_UNZIP_SUM]->store(
|
|
info->unzip_sum, true));
|
|
|
|
OK(fields[IDX_BUF_STATS_UNZIP_CUR]->store(
|
|
info->unzip_cur, true));
|
|
|
|
DBUG_RETURN(schema_table_store_record(thd, table));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
This is the function that loops through each buffer pool and fetch buffer
|
|
pool stats to information schema table: I_S_INNODB_BUFFER_POOL_STATS
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buffer_stats_fill_table(
|
|
/*===============================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
int status = 0;
|
|
buf_pool_info_t* pool_info;
|
|
|
|
DBUG_ENTER("i_s_innodb_buffer_fill_general");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* Only allow the PROCESS privilege holder to access the stats */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
pool_info = (buf_pool_info_t*) ut_zalloc_nokey(
|
|
srv_buf_pool_instances * sizeof *pool_info);
|
|
|
|
/* Walk through each buffer pool */
|
|
for (ulint i = 0; i < srv_buf_pool_instances; i++) {
|
|
buf_pool_t* buf_pool;
|
|
|
|
buf_pool = buf_pool_from_array(i);
|
|
|
|
/* Fetch individual buffer pool info */
|
|
buf_stats_get_pool_info(buf_pool, i, pool_info);
|
|
|
|
status = i_s_innodb_stats_fill(thd, tables, &pool_info[i]);
|
|
|
|
/* If something goes wrong, break and return */
|
|
if (status) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
ut_free(pool_info);
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_BUFFER_POOL_STATS.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buffer_pool_stats_init(
|
|
/*==============================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("i_s_innodb_buffer_pool_stats_init");
|
|
|
|
schema = reinterpret_cast<ST_SCHEMA_TABLE*>(p);
|
|
|
|
schema->fields_info = i_s_innodb_buffer_stats_fields_info;
|
|
schema->fill_table = i_s_innodb_buffer_stats_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_buffer_stats =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_BUFFER_POOL_STATS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB Buffer Pool Statistics Information "),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_innodb_buffer_pool_stats_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/* Fields of the dynamic table INNODB_BUFFER_POOL_PAGE. */
|
|
static ST_FIELD_INFO i_s_innodb_buffer_page_fields_info[] =
|
|
{
|
|
#define IDX_BUFFER_POOL_ID 0
|
|
{STRUCT_FLD(field_name, "POOL_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_BLOCK_ID 1
|
|
{STRUCT_FLD(field_name, "BLOCK_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_SPACE 2
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_NUM 3
|
|
{STRUCT_FLD(field_name, "PAGE_NUMBER"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_TYPE 4
|
|
{STRUCT_FLD(field_name, "PAGE_TYPE"),
|
|
STRUCT_FLD(field_length, 64),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_FLUSH_TYPE 5
|
|
{STRUCT_FLD(field_name, "FLUSH_TYPE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_FIX_COUNT 6
|
|
{STRUCT_FLD(field_name, "FIX_COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_HASHED 7
|
|
{STRUCT_FLD(field_name, "IS_HASHED"),
|
|
STRUCT_FLD(field_length, 3),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_NEWEST_MOD 8
|
|
{STRUCT_FLD(field_name, "NEWEST_MODIFICATION"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_OLDEST_MOD 9
|
|
{STRUCT_FLD(field_name, "OLDEST_MODIFICATION"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_ACCESS_TIME 10
|
|
{STRUCT_FLD(field_name, "ACCESS_TIME"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_TABLE_NAME 11
|
|
{STRUCT_FLD(field_name, "TABLE_NAME"),
|
|
STRUCT_FLD(field_length, 1024),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_INDEX_NAME 12
|
|
{STRUCT_FLD(field_name, "INDEX_NAME"),
|
|
STRUCT_FLD(field_length, 1024),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_NUM_RECS 13
|
|
{STRUCT_FLD(field_name, "NUMBER_RECORDS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_DATA_SIZE 14
|
|
{STRUCT_FLD(field_name, "DATA_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_ZIP_SIZE 15
|
|
{STRUCT_FLD(field_name, "COMPRESSED_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_STATE 16
|
|
{STRUCT_FLD(field_name, "PAGE_STATE"),
|
|
STRUCT_FLD(field_length, 64),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_IO_FIX 17
|
|
{STRUCT_FLD(field_name, "IO_FIX"),
|
|
STRUCT_FLD(field_length, 64),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_IS_OLD 18
|
|
{STRUCT_FLD(field_name, "IS_OLD"),
|
|
STRUCT_FLD(field_length, 3),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUFFER_PAGE_FREE_CLOCK 19
|
|
{STRUCT_FLD(field_name, "FREE_PAGE_CLOCK"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill Information Schema table INNODB_BUFFER_PAGE with information
|
|
cached in the buf_page_info_t array
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buffer_page_fill(
|
|
/*========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
const buf_page_info_t* info_array, /*!< in: array cached page
|
|
info */
|
|
ulint num_page) /*!< in: number of page info
|
|
cached */
|
|
{
|
|
TABLE* table;
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_innodb_buffer_page_fill");
|
|
|
|
table = tables->table;
|
|
|
|
fields = table->field;
|
|
|
|
/* Iterate through the cached array and fill the I_S table rows */
|
|
for (ulint i = 0; i < num_page; i++) {
|
|
const buf_page_info_t* page_info;
|
|
char table_name[MAX_FULL_NAME_LEN + 1];
|
|
const char* table_name_end = NULL;
|
|
const char* state_str;
|
|
enum buf_page_state state;
|
|
|
|
page_info = info_array + i;
|
|
|
|
state_str = NULL;
|
|
|
|
OK(fields[IDX_BUFFER_POOL_ID]->store(
|
|
page_info->pool_id, true));
|
|
|
|
OK(fields[IDX_BUFFER_BLOCK_ID]->store(
|
|
page_info->block_id, true));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_SPACE]->store(
|
|
page_info->space_id, true));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_NUM]->store(
|
|
page_info->page_num, true));
|
|
|
|
OK(field_store_string(
|
|
fields[IDX_BUFFER_PAGE_TYPE],
|
|
i_s_page_type[page_info->page_type].type_str));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_FLUSH_TYPE]->store(
|
|
page_info->flush_type));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_FIX_COUNT]->store(
|
|
page_info->fix_count));
|
|
|
|
if (page_info->hashed) {
|
|
OK(field_store_string(
|
|
fields[IDX_BUFFER_PAGE_HASHED], "YES"));
|
|
} else {
|
|
OK(field_store_string(
|
|
fields[IDX_BUFFER_PAGE_HASHED], "NO"));
|
|
}
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_NEWEST_MOD]->store(
|
|
page_info->newest_mod, true));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_OLDEST_MOD]->store(
|
|
page_info->oldest_mod, true));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_ACCESS_TIME]->store(
|
|
page_info->access_time));
|
|
|
|
fields[IDX_BUFFER_PAGE_TABLE_NAME]->set_null();
|
|
|
|
fields[IDX_BUFFER_PAGE_INDEX_NAME]->set_null();
|
|
|
|
/* If this is an index page, fetch the index name
|
|
and table name */
|
|
if (page_info->page_type == I_S_PAGE_TYPE_INDEX) {
|
|
const dict_index_t* index;
|
|
|
|
mutex_enter(&dict_sys->mutex);
|
|
index = dict_index_get_if_in_cache_low(
|
|
page_info->index_id);
|
|
|
|
if (index) {
|
|
|
|
table_name_end = innobase_convert_name(
|
|
table_name, sizeof(table_name),
|
|
index->table_name,
|
|
strlen(index->table_name),
|
|
thd);
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_TABLE_NAME]->store(
|
|
table_name,
|
|
static_cast<size_t>(table_name_end - table_name),
|
|
system_charset_info));
|
|
fields[IDX_BUFFER_PAGE_TABLE_NAME]->set_notnull();
|
|
|
|
OK(field_store_index_name(
|
|
fields[IDX_BUFFER_PAGE_INDEX_NAME],
|
|
index->name));
|
|
}
|
|
|
|
mutex_exit(&dict_sys->mutex);
|
|
}
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_NUM_RECS]->store(
|
|
page_info->num_recs, true));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_DATA_SIZE]->store(
|
|
page_info->data_size, true));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_ZIP_SIZE]->store(
|
|
page_info->zip_ssize
|
|
? (UNIV_ZIP_SIZE_MIN >> 1) << page_info->zip_ssize
|
|
: 0, true));
|
|
|
|
#if BUF_PAGE_STATE_BITS > 3
|
|
# error "BUF_PAGE_STATE_BITS > 3, please ensure that all 1<<BUF_PAGE_STATE_BITS values are checked for"
|
|
#endif
|
|
state = static_cast<enum buf_page_state>(page_info->page_state);
|
|
|
|
switch (state) {
|
|
/* First three states are for compression pages and
|
|
are not states we would get as we scan pages through
|
|
buffer blocks */
|
|
case BUF_BLOCK_POOL_WATCH:
|
|
case BUF_BLOCK_ZIP_PAGE:
|
|
case BUF_BLOCK_ZIP_DIRTY:
|
|
state_str = NULL;
|
|
break;
|
|
case BUF_BLOCK_NOT_USED:
|
|
state_str = "NOT_USED";
|
|
break;
|
|
case BUF_BLOCK_READY_FOR_USE:
|
|
state_str = "READY_FOR_USE";
|
|
break;
|
|
case BUF_BLOCK_FILE_PAGE:
|
|
state_str = "FILE_PAGE";
|
|
break;
|
|
case BUF_BLOCK_MEMORY:
|
|
state_str = "MEMORY";
|
|
break;
|
|
case BUF_BLOCK_REMOVE_HASH:
|
|
state_str = "REMOVE_HASH";
|
|
break;
|
|
};
|
|
|
|
OK(field_store_string(fields[IDX_BUFFER_PAGE_STATE],
|
|
state_str));
|
|
|
|
switch (page_info->io_fix) {
|
|
case BUF_IO_NONE:
|
|
OK(field_store_string(fields[IDX_BUFFER_PAGE_IO_FIX],
|
|
"IO_NONE"));
|
|
break;
|
|
case BUF_IO_READ:
|
|
OK(field_store_string(fields[IDX_BUFFER_PAGE_IO_FIX],
|
|
"IO_READ"));
|
|
break;
|
|
case BUF_IO_WRITE:
|
|
OK(field_store_string(fields[IDX_BUFFER_PAGE_IO_FIX],
|
|
"IO_WRITE"));
|
|
break;
|
|
case BUF_IO_PIN:
|
|
OK(field_store_string(fields[IDX_BUFFER_PAGE_IO_FIX],
|
|
"IO_PIN"));
|
|
break;
|
|
}
|
|
|
|
OK(field_store_string(fields[IDX_BUFFER_PAGE_IS_OLD],
|
|
(page_info->is_old) ? "YES" : "NO"));
|
|
|
|
OK(fields[IDX_BUFFER_PAGE_FREE_CLOCK]->store(
|
|
page_info->freed_page_clock, true));
|
|
|
|
if (schema_table_store_record(thd, table)) {
|
|
DBUG_RETURN(1);
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Set appropriate page type to a buf_page_info_t structure */
|
|
static
|
|
void
|
|
i_s_innodb_set_page_type(
|
|
/*=====================*/
|
|
buf_page_info_t*page_info, /*!< in/out: structure to fill with
|
|
scanned info */
|
|
ulint page_type, /*!< in: page type */
|
|
const byte* frame) /*!< in: buffer frame */
|
|
{
|
|
if (fil_page_type_is_index(page_type)) {
|
|
const page_t* page = (const page_t*) frame;
|
|
|
|
page_info->index_id = btr_page_get_index_id(page);
|
|
|
|
/* FIL_PAGE_INDEX and FIL_PAGE_RTREE are a bit special,
|
|
their values are defined as 17855 and 17854, so we cannot
|
|
use them to index into i_s_page_type[] array, its array index
|
|
in the i_s_page_type[] array is I_S_PAGE_TYPE_INDEX
|
|
(1) for index pages or I_S_PAGE_TYPE_IBUF for
|
|
change buffer index pages */
|
|
if (page_info->index_id
|
|
== static_cast<index_id_t>(DICT_IBUF_ID_MIN
|
|
+ IBUF_SPACE_ID)) {
|
|
page_info->page_type = I_S_PAGE_TYPE_IBUF;
|
|
} else if (page_type == FIL_PAGE_RTREE) {
|
|
page_info->page_type = I_S_PAGE_TYPE_RTREE;
|
|
} else {
|
|
page_info->page_type = I_S_PAGE_TYPE_INDEX;
|
|
}
|
|
|
|
page_info->data_size = (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));
|
|
|
|
page_info->num_recs = page_get_n_recs(page);
|
|
} else if (page_type > FIL_PAGE_TYPE_LAST) {
|
|
/* Encountered an unknown page type */
|
|
page_info->page_type = I_S_PAGE_TYPE_UNKNOWN;
|
|
} else {
|
|
/* Make sure we get the right index into the
|
|
i_s_page_type[] array */
|
|
ut_a(page_type == i_s_page_type[page_type].type_value);
|
|
|
|
page_info->page_type = page_type;
|
|
}
|
|
|
|
if (page_info->page_type == FIL_PAGE_TYPE_ZBLOB
|
|
|| page_info->page_type == FIL_PAGE_TYPE_ZBLOB2) {
|
|
page_info->page_num = mach_read_from_4(
|
|
frame + FIL_PAGE_OFFSET);
|
|
page_info->space_id = mach_read_from_4(
|
|
frame + FIL_PAGE_ARCH_LOG_NO_OR_SPACE_ID);
|
|
}
|
|
}
|
|
/*******************************************************************//**
|
|
Scans pages in the buffer cache, and collect their general information
|
|
into the buf_page_info_t array which is zero-filled. So any fields
|
|
that are not initialized in the function will default to 0 */
|
|
static
|
|
void
|
|
i_s_innodb_buffer_page_get_info(
|
|
/*============================*/
|
|
const buf_page_t*bpage, /*!< in: buffer pool page to scan */
|
|
ulint pool_id, /*!< in: buffer pool id */
|
|
ulint pos, /*!< in: buffer block position in
|
|
buffer pool or in the LRU list */
|
|
buf_page_info_t*page_info) /*!< in: zero filled info structure;
|
|
out: structure filled with scanned
|
|
info */
|
|
{
|
|
ut_ad(pool_id < MAX_BUFFER_POOLS);
|
|
|
|
page_info->pool_id = pool_id;
|
|
|
|
page_info->block_id = pos;
|
|
|
|
page_info->page_state = buf_page_get_state(bpage);
|
|
|
|
/* Only fetch information for buffers that map to a tablespace,
|
|
that is, buffer page with state BUF_BLOCK_ZIP_PAGE,
|
|
BUF_BLOCK_ZIP_DIRTY or BUF_BLOCK_FILE_PAGE */
|
|
if (buf_page_in_file(bpage)) {
|
|
const byte* frame;
|
|
ulint page_type;
|
|
|
|
page_info->space_id = bpage->id.space();
|
|
|
|
page_info->page_num = bpage->id.page_no();
|
|
|
|
page_info->flush_type = bpage->flush_type;
|
|
|
|
page_info->fix_count = bpage->buf_fix_count;
|
|
|
|
page_info->newest_mod = bpage->newest_modification;
|
|
|
|
page_info->oldest_mod = bpage->oldest_modification;
|
|
|
|
page_info->access_time = bpage->access_time;
|
|
|
|
page_info->zip_ssize = bpage->zip.ssize;
|
|
|
|
page_info->io_fix = bpage->io_fix;
|
|
|
|
page_info->is_old = bpage->old;
|
|
|
|
page_info->freed_page_clock = bpage->freed_page_clock;
|
|
|
|
switch (buf_page_get_io_fix(bpage)) {
|
|
case BUF_IO_NONE:
|
|
case BUF_IO_WRITE:
|
|
case BUF_IO_PIN:
|
|
break;
|
|
case BUF_IO_READ:
|
|
page_info->page_type = I_S_PAGE_TYPE_UNKNOWN;
|
|
return;
|
|
}
|
|
|
|
if (page_info->page_state == BUF_BLOCK_FILE_PAGE) {
|
|
const buf_block_t*block;
|
|
|
|
block = reinterpret_cast<const buf_block_t*>(bpage);
|
|
frame = block->frame;
|
|
page_info->hashed = (block->index != NULL);
|
|
} else {
|
|
ut_ad(page_info->zip_ssize);
|
|
frame = bpage->zip.data;
|
|
}
|
|
|
|
page_type = fil_page_get_type(frame);
|
|
|
|
i_s_innodb_set_page_type(page_info, page_type, frame);
|
|
} else {
|
|
page_info->page_type = I_S_PAGE_TYPE_UNKNOWN;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
This is the function that goes through each block of the buffer pool
|
|
and fetch information to information schema tables: INNODB_BUFFER_PAGE.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_fill_buffer_pool(
|
|
/*========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
buf_pool_t* buf_pool, /*!< in: buffer pool to scan */
|
|
const ulint pool_id) /*!< in: buffer pool id */
|
|
{
|
|
int status = 0;
|
|
mem_heap_t* heap;
|
|
|
|
DBUG_ENTER("i_s_innodb_fill_buffer_pool");
|
|
|
|
heap = mem_heap_create(10000);
|
|
|
|
/* Go through each chunk of buffer pool. Currently, we only
|
|
have one single chunk for each buffer pool */
|
|
for (ulint n = 0;
|
|
n < ut_min(buf_pool->n_chunks, buf_pool->n_chunks_new); n++) {
|
|
const buf_block_t* block;
|
|
ulint n_blocks;
|
|
buf_page_info_t* info_buffer;
|
|
ulint num_page;
|
|
ulint mem_size;
|
|
ulint chunk_size;
|
|
ulint num_to_process = 0;
|
|
ulint block_id = 0;
|
|
|
|
/* Get buffer block of the nth chunk */
|
|
block = buf_get_nth_chunk_block(buf_pool, n, &chunk_size);
|
|
num_page = 0;
|
|
|
|
while (chunk_size > 0) {
|
|
/* we cache maximum MAX_BUF_INFO_CACHED number of
|
|
buffer page info */
|
|
num_to_process = ut_min(chunk_size,
|
|
(ulint)MAX_BUF_INFO_CACHED);
|
|
|
|
mem_size = num_to_process * sizeof(buf_page_info_t);
|
|
|
|
/* For each chunk, we'll pre-allocate information
|
|
structures to cache the page information read from
|
|
the buffer pool. Doing so before obtain any mutex */
|
|
info_buffer = (buf_page_info_t*) mem_heap_zalloc(
|
|
heap, mem_size);
|
|
|
|
/* Obtain appropriate mutexes. Since this is diagnostic
|
|
buffer pool info printout, we are not required to
|
|
preserve the overall consistency, so we can
|
|
release mutex periodically */
|
|
buf_pool_mutex_enter(buf_pool);
|
|
|
|
/* GO through each block in the chunk */
|
|
for (n_blocks = num_to_process; n_blocks--; block++) {
|
|
i_s_innodb_buffer_page_get_info(
|
|
&block->page, pool_id, block_id,
|
|
info_buffer + num_page);
|
|
block_id++;
|
|
num_page++;
|
|
}
|
|
|
|
buf_pool_mutex_exit(buf_pool);
|
|
|
|
/* Fill in information schema table with information
|
|
just collected from the buffer chunk scan */
|
|
status = i_s_innodb_buffer_page_fill(
|
|
thd, tables, info_buffer,
|
|
num_page);
|
|
|
|
/* If something goes wrong, break and return */
|
|
if (status) {
|
|
break;
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
chunk_size -= num_to_process;
|
|
num_page = 0;
|
|
}
|
|
}
|
|
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill page information for pages in InnoDB buffer pool to the
|
|
dynamic table INFORMATION_SCHEMA.INNODB_BUFFER_PAGE
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buffer_page_fill_table(
|
|
/*==============================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
int status = 0;
|
|
|
|
DBUG_ENTER("i_s_innodb_buffer_page_fill_table");
|
|
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* Walk through each buffer pool */
|
|
for (ulint i = 0; i < srv_buf_pool_instances; i++) {
|
|
buf_pool_t* buf_pool;
|
|
|
|
buf_pool = buf_pool_from_array(i);
|
|
|
|
/* Fetch information from pages in this buffer pool,
|
|
and fill the corresponding I_S table */
|
|
status = i_s_innodb_fill_buffer_pool(thd, tables, buf_pool, i);
|
|
|
|
/* If something wrong, break and return */
|
|
if (status) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_BUFFER_PAGE.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buffer_page_init(
|
|
/*========================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("i_s_innodb_buffer_page_init");
|
|
|
|
schema = reinterpret_cast<ST_SCHEMA_TABLE*>(p);
|
|
|
|
schema->fields_info = i_s_innodb_buffer_page_fields_info;
|
|
schema->fill_table = i_s_innodb_buffer_page_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_buffer_page =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_BUFFER_PAGE"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB Buffer Page Information"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_innodb_buffer_page_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
static ST_FIELD_INFO i_s_innodb_buf_page_lru_fields_info[] =
|
|
{
|
|
#define IDX_BUF_LRU_POOL_ID 0
|
|
{STRUCT_FLD(field_name, "POOL_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_POS 1
|
|
{STRUCT_FLD(field_name, "LRU_POSITION"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_SPACE 2
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_NUM 3
|
|
{STRUCT_FLD(field_name, "PAGE_NUMBER"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_TYPE 4
|
|
{STRUCT_FLD(field_name, "PAGE_TYPE"),
|
|
STRUCT_FLD(field_length, 64),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_FLUSH_TYPE 5
|
|
{STRUCT_FLD(field_name, "FLUSH_TYPE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_FIX_COUNT 6
|
|
{STRUCT_FLD(field_name, "FIX_COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_HASHED 7
|
|
{STRUCT_FLD(field_name, "IS_HASHED"),
|
|
STRUCT_FLD(field_length, 3),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_NEWEST_MOD 8
|
|
{STRUCT_FLD(field_name, "NEWEST_MODIFICATION"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_OLDEST_MOD 9
|
|
{STRUCT_FLD(field_name, "OLDEST_MODIFICATION"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_ACCESS_TIME 10
|
|
{STRUCT_FLD(field_name, "ACCESS_TIME"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_TABLE_NAME 11
|
|
{STRUCT_FLD(field_name, "TABLE_NAME"),
|
|
STRUCT_FLD(field_length, 1024),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_INDEX_NAME 12
|
|
{STRUCT_FLD(field_name, "INDEX_NAME"),
|
|
STRUCT_FLD(field_length, 1024),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_NUM_RECS 13
|
|
{STRUCT_FLD(field_name, "NUMBER_RECORDS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_DATA_SIZE 14
|
|
{STRUCT_FLD(field_name, "DATA_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_ZIP_SIZE 15
|
|
{STRUCT_FLD(field_name, "COMPRESSED_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_STATE 16
|
|
{STRUCT_FLD(field_name, "COMPRESSED"),
|
|
STRUCT_FLD(field_length, 3),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_IO_FIX 17
|
|
{STRUCT_FLD(field_name, "IO_FIX"),
|
|
STRUCT_FLD(field_length, 64),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_IS_OLD 18
|
|
{STRUCT_FLD(field_name, "IS_OLD"),
|
|
STRUCT_FLD(field_length, 3),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define IDX_BUF_LRU_PAGE_FREE_CLOCK 19
|
|
{STRUCT_FLD(field_name, "FREE_PAGE_CLOCK"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Fill Information Schema table INNODB_BUFFER_PAGE_LRU with information
|
|
cached in the buf_page_info_t array
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buf_page_lru_fill(
|
|
/*=========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
const buf_page_info_t* info_array, /*!< in: array cached page
|
|
info */
|
|
ulint num_page) /*!< in: number of page info
|
|
cached */
|
|
{
|
|
TABLE* table;
|
|
Field** fields;
|
|
mem_heap_t* heap;
|
|
|
|
DBUG_ENTER("i_s_innodb_buf_page_lru_fill");
|
|
|
|
table = tables->table;
|
|
|
|
fields = table->field;
|
|
|
|
heap = mem_heap_create(1000);
|
|
|
|
/* Iterate through the cached array and fill the I_S table rows */
|
|
for (ulint i = 0; i < num_page; i++) {
|
|
const buf_page_info_t* page_info;
|
|
char table_name[MAX_FULL_NAME_LEN + 1];
|
|
const char* table_name_end = NULL;
|
|
const char* state_str;
|
|
enum buf_page_state state;
|
|
|
|
state_str = NULL;
|
|
|
|
page_info = info_array + i;
|
|
|
|
OK(fields[IDX_BUF_LRU_POOL_ID]->store(
|
|
page_info->pool_id, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_POS]->store(
|
|
page_info->block_id, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_SPACE]->store(
|
|
page_info->space_id, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_NUM]->store(
|
|
page_info->page_num, true));
|
|
|
|
OK(field_store_string(
|
|
fields[IDX_BUF_LRU_PAGE_TYPE],
|
|
i_s_page_type[page_info->page_type].type_str));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_FLUSH_TYPE]->store(
|
|
page_info->flush_type, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_FIX_COUNT]->store(
|
|
page_info->fix_count, true));
|
|
|
|
if (page_info->hashed) {
|
|
OK(field_store_string(
|
|
fields[IDX_BUF_LRU_PAGE_HASHED], "YES"));
|
|
} else {
|
|
OK(field_store_string(
|
|
fields[IDX_BUF_LRU_PAGE_HASHED], "NO"));
|
|
}
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_NEWEST_MOD]->store(
|
|
page_info->newest_mod, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_OLDEST_MOD]->store(
|
|
page_info->oldest_mod, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_ACCESS_TIME]->store(
|
|
page_info->access_time, true));
|
|
|
|
fields[IDX_BUF_LRU_PAGE_TABLE_NAME]->set_null();
|
|
|
|
fields[IDX_BUF_LRU_PAGE_INDEX_NAME]->set_null();
|
|
|
|
/* If this is an index page, fetch the index name
|
|
and table name */
|
|
if (page_info->page_type == I_S_PAGE_TYPE_INDEX) {
|
|
const dict_index_t* index;
|
|
|
|
mutex_enter(&dict_sys->mutex);
|
|
index = dict_index_get_if_in_cache_low(
|
|
page_info->index_id);
|
|
|
|
if (index) {
|
|
|
|
table_name_end = innobase_convert_name(
|
|
table_name, sizeof(table_name),
|
|
index->table_name,
|
|
strlen(index->table_name),
|
|
thd);
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_TABLE_NAME]->store(
|
|
table_name,
|
|
static_cast<size_t>(table_name_end - table_name),
|
|
system_charset_info));
|
|
fields[IDX_BUF_LRU_PAGE_TABLE_NAME]->set_notnull();
|
|
|
|
OK(field_store_index_name(
|
|
fields[IDX_BUF_LRU_PAGE_INDEX_NAME],
|
|
index->name));
|
|
}
|
|
|
|
mutex_exit(&dict_sys->mutex);
|
|
}
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_NUM_RECS]->store(
|
|
page_info->num_recs, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_DATA_SIZE]->store(
|
|
page_info->data_size, true));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_ZIP_SIZE]->store(
|
|
page_info->zip_ssize ?
|
|
512 << page_info->zip_ssize : 0, true));
|
|
|
|
state = static_cast<enum buf_page_state>(page_info->page_state);
|
|
|
|
switch (state) {
|
|
/* Compressed page */
|
|
case BUF_BLOCK_ZIP_PAGE:
|
|
case BUF_BLOCK_ZIP_DIRTY:
|
|
state_str = "YES";
|
|
break;
|
|
/* Uncompressed page */
|
|
case BUF_BLOCK_FILE_PAGE:
|
|
state_str = "NO";
|
|
break;
|
|
/* We should not see following states */
|
|
case BUF_BLOCK_POOL_WATCH:
|
|
case BUF_BLOCK_READY_FOR_USE:
|
|
case BUF_BLOCK_NOT_USED:
|
|
case BUF_BLOCK_MEMORY:
|
|
case BUF_BLOCK_REMOVE_HASH:
|
|
state_str = NULL;
|
|
break;
|
|
};
|
|
|
|
OK(field_store_string(fields[IDX_BUF_LRU_PAGE_STATE],
|
|
state_str));
|
|
|
|
switch (page_info->io_fix) {
|
|
case BUF_IO_NONE:
|
|
OK(field_store_string(fields[IDX_BUF_LRU_PAGE_IO_FIX],
|
|
"IO_NONE"));
|
|
break;
|
|
case BUF_IO_READ:
|
|
OK(field_store_string(fields[IDX_BUF_LRU_PAGE_IO_FIX],
|
|
"IO_READ"));
|
|
break;
|
|
case BUF_IO_WRITE:
|
|
OK(field_store_string(fields[IDX_BUF_LRU_PAGE_IO_FIX],
|
|
"IO_WRITE"));
|
|
break;
|
|
}
|
|
|
|
OK(field_store_string(fields[IDX_BUF_LRU_PAGE_IS_OLD],
|
|
(page_info->is_old) ? "YES" : "NO"));
|
|
|
|
OK(fields[IDX_BUF_LRU_PAGE_FREE_CLOCK]->store(
|
|
page_info->freed_page_clock, true));
|
|
|
|
if (schema_table_store_record(thd, table)) {
|
|
mem_heap_free(heap);
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
}
|
|
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
This is the function that goes through buffer pool's LRU list
|
|
and fetch information to INFORMATION_SCHEMA.INNODB_BUFFER_PAGE_LRU.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_fill_buffer_lru(
|
|
/*=======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
buf_pool_t* buf_pool, /*!< in: buffer pool to scan */
|
|
const ulint pool_id) /*!< in: buffer pool id */
|
|
{
|
|
int status = 0;
|
|
buf_page_info_t* info_buffer;
|
|
ulint lru_pos = 0;
|
|
const buf_page_t* bpage;
|
|
ulint lru_len;
|
|
|
|
DBUG_ENTER("i_s_innodb_fill_buffer_lru");
|
|
|
|
/* Obtain buf_pool mutex before allocate info_buffer, since
|
|
UT_LIST_GET_LEN(buf_pool->LRU) could change */
|
|
buf_pool_mutex_enter(buf_pool);
|
|
|
|
lru_len = UT_LIST_GET_LEN(buf_pool->LRU);
|
|
|
|
/* Print error message if malloc fail */
|
|
info_buffer = (buf_page_info_t*) my_malloc(
|
|
lru_len * sizeof *info_buffer, MYF(MY_WME));
|
|
/* JAN: TODO: MySQL 5.7 PSI
|
|
info_buffer = (buf_page_info_t*) my_malloc(PSI_INSTRUMENT_ME,
|
|
lru_len * sizeof *info_buffer, MYF(MY_WME));
|
|
*/
|
|
|
|
if (!info_buffer) {
|
|
status = 1;
|
|
goto exit;
|
|
}
|
|
|
|
memset(info_buffer, 0, lru_len * sizeof *info_buffer);
|
|
|
|
/* Walk through Pool's LRU list and print the buffer page
|
|
information */
|
|
bpage = UT_LIST_GET_LAST(buf_pool->LRU);
|
|
|
|
while (bpage != NULL) {
|
|
/* Use the same function that collect buffer info for
|
|
INNODB_BUFFER_PAGE to get buffer page info */
|
|
i_s_innodb_buffer_page_get_info(bpage, pool_id, lru_pos,
|
|
(info_buffer + lru_pos));
|
|
|
|
bpage = UT_LIST_GET_PREV(LRU, bpage);
|
|
|
|
lru_pos++;
|
|
}
|
|
|
|
ut_ad(lru_pos == lru_len);
|
|
ut_ad(lru_pos == UT_LIST_GET_LEN(buf_pool->LRU));
|
|
|
|
exit:
|
|
buf_pool_mutex_exit(buf_pool);
|
|
|
|
if (info_buffer) {
|
|
status = i_s_innodb_buf_page_lru_fill(
|
|
thd, tables, info_buffer, lru_len);
|
|
|
|
my_free(info_buffer);
|
|
}
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Fill page information for pages in InnoDB buffer pool to the
|
|
dynamic table INFORMATION_SCHEMA.INNODB_BUFFER_PAGE_LRU
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buf_page_lru_fill_table(
|
|
/*===============================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (ignored) */
|
|
{
|
|
int status = 0;
|
|
|
|
DBUG_ENTER("i_s_innodb_buf_page_lru_fill_table");
|
|
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to any users that do not hold PROCESS_ACL */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* Walk through each buffer pool */
|
|
for (ulint i = 0; i < srv_buf_pool_instances; i++) {
|
|
buf_pool_t* buf_pool;
|
|
|
|
buf_pool = buf_pool_from_array(i);
|
|
|
|
/* Fetch information from pages in this buffer pool's LRU list,
|
|
and fill the corresponding I_S table */
|
|
status = i_s_innodb_fill_buffer_lru(thd, tables, buf_pool, i);
|
|
|
|
/* If something wrong, break and return */
|
|
if (status) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_BUFFER_PAGE_LRU.
|
|
@return 0 on success, 1 on failure */
|
|
static
|
|
int
|
|
i_s_innodb_buffer_page_lru_init(
|
|
/*============================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("i_s_innodb_buffer_page_lru_init");
|
|
|
|
schema = reinterpret_cast<ST_SCHEMA_TABLE*>(p);
|
|
|
|
schema->fields_info = i_s_innodb_buf_page_lru_fields_info;
|
|
schema->fill_table = i_s_innodb_buf_page_lru_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_buffer_page_lru =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_BUFFER_PAGE_LRU"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB Buffer Page in LRU"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, i_s_innodb_buffer_page_lru_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Unbind a dynamic INFORMATION_SCHEMA table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_common_deinit(
|
|
/*==============*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("i_s_common_deinit");
|
|
|
|
/* Do nothing */
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/** SYS_TABLES ***************************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.SYS_TABLES */
|
|
static ST_FIELD_INFO innodb_sys_tables_fields_info[] =
|
|
{
|
|
#define SYS_TABLES_ID 0
|
|
{STRUCT_FLD(field_name, "TABLE_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, MAX_FULL_NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_FLAG 2
|
|
{STRUCT_FLD(field_name, "FLAG"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_NUM_COLUMN 3
|
|
{STRUCT_FLD(field_name, "N_COLS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_SPACE 4
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_FILE_FORMAT 5
|
|
{STRUCT_FLD(field_name, "FILE_FORMAT"),
|
|
STRUCT_FLD(field_length, 10),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_ROW_FORMAT 6
|
|
{STRUCT_FLD(field_name, "ROW_FORMAT"),
|
|
STRUCT_FLD(field_length, 12),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_ZIP_PAGE_SIZE 7
|
|
{STRUCT_FLD(field_name, "ZIP_PAGE_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLES_SPACE_TYPE 8
|
|
{STRUCT_FLD(field_name, "SPACE_TYPE"),
|
|
STRUCT_FLD(field_length, 10),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Populate information_schema.innodb_sys_tables table with information
|
|
from SYS_TABLES.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_tables(
|
|
/*=====================*/
|
|
THD* thd, /*!< in: thread */
|
|
dict_table_t* table, /*!< in: table */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
ulint compact = DICT_TF_GET_COMPACT(table->flags);
|
|
ulint atomic_blobs = DICT_TF_HAS_ATOMIC_BLOBS(
|
|
table->flags);
|
|
const page_size_t& page_size = dict_tf_get_page_size(table->flags);
|
|
const char* file_format;
|
|
const char* row_format;
|
|
const char* space_type;
|
|
|
|
file_format = trx_sys_file_format_id_to_name(atomic_blobs);
|
|
if (!compact) {
|
|
row_format = "Redundant";
|
|
} else if (!atomic_blobs) {
|
|
row_format = "Compact";
|
|
} else if (DICT_TF_GET_ZIP_SSIZE(table->flags)) {
|
|
row_format = "Compressed";
|
|
} else {
|
|
row_format = "Dynamic";
|
|
}
|
|
|
|
if (is_system_tablespace(table->space)) {
|
|
space_type = "System";
|
|
} else {
|
|
space_type = "Single";
|
|
}
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_tables");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(fields[SYS_TABLES_ID]->store(longlong(table->id), TRUE));
|
|
|
|
OK(field_store_string(fields[SYS_TABLES_NAME], table->name.m_name));
|
|
|
|
OK(fields[SYS_TABLES_FLAG]->store(table->flags));
|
|
|
|
OK(fields[SYS_TABLES_NUM_COLUMN]->store(table->n_cols));
|
|
|
|
OK(fields[SYS_TABLES_SPACE]->store(table->space));
|
|
|
|
OK(field_store_string(fields[SYS_TABLES_FILE_FORMAT], file_format));
|
|
|
|
OK(field_store_string(fields[SYS_TABLES_ROW_FORMAT], row_format));
|
|
|
|
OK(fields[SYS_TABLES_ZIP_PAGE_SIZE]->store(static_cast<double>(
|
|
page_size.is_compressed()
|
|
? page_size.physical()
|
|
: 0)));
|
|
|
|
OK(field_store_string(fields[SYS_TABLES_SPACE_TYPE], space_type));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to go through each record in SYS_TABLES table, and fill the
|
|
information_schema.innodb_sys_tables table with related table information
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_tables_fill_table(
|
|
/*======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_tables_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_TABLES);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
dict_table_t* table_rec;
|
|
|
|
/* Create and populate a dict_table_t structure with
|
|
information from SYS_TABLES row */
|
|
err_msg = dict_process_sys_tables_rec_and_mtr_commit(
|
|
heap, rec, &table_rec,
|
|
DICT_TABLE_LOAD_FROM_RECORD, &mtr);
|
|
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_tables(thd, table_rec, tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
/* Since dict_process_sys_tables_rec_and_mtr_commit()
|
|
is called with DICT_TABLE_LOAD_FROM_RECORD, the table_rec
|
|
is created in dict_process_sys_tables_rec(), we will
|
|
need to free it */
|
|
if (table_rec) {
|
|
dict_mem_table_free(table_rec);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_tables
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_tables_init(
|
|
/*===================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_tables_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_tables_fields_info;
|
|
schema->fill_table = i_s_sys_tables_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_tables =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_TABLES"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_TABLES"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_tables_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_TABLESTATS ***********************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.SYS_TABLESTATS */
|
|
static ST_FIELD_INFO innodb_sys_tablestats_fields_info[] =
|
|
{
|
|
#define SYS_TABLESTATS_ID 0
|
|
{STRUCT_FLD(field_name, "TABLE_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_INIT 2
|
|
{STRUCT_FLD(field_name, "STATS_INITIALIZED"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_NROW 3
|
|
{STRUCT_FLD(field_name, "NUM_ROWS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_CLUST_SIZE 4
|
|
{STRUCT_FLD(field_name, "CLUST_INDEX_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_INDEX_SIZE 5
|
|
{STRUCT_FLD(field_name, "OTHER_INDEX_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_MODIFIED 6
|
|
{STRUCT_FLD(field_name, "MODIFIED_COUNTER"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_AUTONINC 7
|
|
{STRUCT_FLD(field_name, "AUTOINC"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESTATS_TABLE_REF_COUNT 8
|
|
{STRUCT_FLD(field_name, "REF_COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/** Populate information_schema.innodb_sys_tablestats table with information
|
|
from SYS_TABLES.
|
|
@param[in] thd thread ID
|
|
@param[in,out] table table
|
|
@param[in] ref_count table reference count
|
|
@param[in,out] table_to_fill fill this table
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_tablestats(
|
|
THD* thd,
|
|
dict_table_t* table,
|
|
ulint ref_count,
|
|
TABLE* table_to_fill)
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_tablestats");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(fields[SYS_TABLESTATS_ID]->store(longlong(table->id), TRUE));
|
|
|
|
OK(field_store_string(fields[SYS_TABLESTATS_NAME],
|
|
table->name.m_name));
|
|
|
|
dict_table_stats_lock(table, RW_S_LATCH);
|
|
|
|
if (table->stat_initialized) {
|
|
OK(field_store_string(fields[SYS_TABLESTATS_INIT],
|
|
"Initialized"));
|
|
|
|
OK(fields[SYS_TABLESTATS_NROW]->store(table->stat_n_rows,
|
|
true));
|
|
|
|
OK(fields[SYS_TABLESTATS_CLUST_SIZE]->store(
|
|
table->stat_clustered_index_size, true));
|
|
|
|
OK(fields[SYS_TABLESTATS_INDEX_SIZE]->store(
|
|
table->stat_sum_of_other_index_sizes, true));
|
|
|
|
OK(fields[SYS_TABLESTATS_MODIFIED]->store(
|
|
table->stat_modified_counter, true));
|
|
} else {
|
|
OK(field_store_string(fields[SYS_TABLESTATS_INIT],
|
|
"Uninitialized"));
|
|
|
|
OK(fields[SYS_TABLESTATS_NROW]->store(0, true));
|
|
|
|
OK(fields[SYS_TABLESTATS_CLUST_SIZE]->store(0, true));
|
|
|
|
OK(fields[SYS_TABLESTATS_INDEX_SIZE]->store(0, true));
|
|
|
|
OK(fields[SYS_TABLESTATS_MODIFIED]->store(0, true));
|
|
}
|
|
|
|
dict_table_stats_unlock(table, RW_S_LATCH);
|
|
|
|
OK(fields[SYS_TABLESTATS_AUTONINC]->store(table->autoinc, true));
|
|
|
|
OK(fields[SYS_TABLESTATS_TABLE_REF_COUNT]->store(static_cast<double>(ref_count)));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Function to go through each record in SYS_TABLES table, and fill the
|
|
information_schema.innodb_sys_tablestats table with table statistics
|
|
related information
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_tables_fill_table_stats(
|
|
/*============================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_tables_fill_table_stats");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_TABLES);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
dict_table_t* table_rec;
|
|
ulint ref_count;
|
|
|
|
/* Fetch the dict_table_t structure corresponding to
|
|
this SYS_TABLES record */
|
|
err_msg = dict_process_sys_tables_rec_and_mtr_commit(
|
|
heap, rec, &table_rec,
|
|
DICT_TABLE_LOAD_FROM_CACHE, &mtr);
|
|
|
|
if (table_rec != NULL) {
|
|
ut_ad(err_msg == NULL);
|
|
|
|
ref_count = table_rec->get_ref_count();
|
|
|
|
/* Protect the dict_table_t object by incrementing
|
|
the reference count. */
|
|
table_rec->acquire();
|
|
}
|
|
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
DBUG_EXECUTE_IF("test_sys_tablestats", {
|
|
if (strcmp("test/t1", table_rec->name.m_name) == 0 ) {
|
|
DEBUG_SYNC_C("dict_table_not_protected");
|
|
}});
|
|
|
|
if (table_rec != NULL) {
|
|
i_s_dict_fill_sys_tablestats(thd, table_rec, ref_count,
|
|
tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
|
|
if (table_rec != NULL) {
|
|
table_rec->release();
|
|
}
|
|
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_tablestats
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_tablestats_init(
|
|
/*=======================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_tablestats_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_tablestats_fields_info;
|
|
schema->fill_table = i_s_sys_tables_fill_table_stats;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_tablestats =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_TABLESTATS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_TABLESTATS"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_tablestats_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_INDEXES **************************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.SYS_INDEXES */
|
|
static ST_FIELD_INFO innodb_sysindex_fields_info[] =
|
|
{
|
|
#define SYS_INDEX_ID 0
|
|
{STRUCT_FLD(field_name, "INDEX_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_INDEX_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_INDEX_TABLE_ID 2
|
|
{STRUCT_FLD(field_name, "TABLE_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_INDEX_TYPE 3
|
|
{STRUCT_FLD(field_name, "TYPE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_INDEX_NUM_FIELDS 4
|
|
{STRUCT_FLD(field_name, "N_FIELDS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_INDEX_PAGE_NO 5
|
|
{STRUCT_FLD(field_name, "PAGE_NO"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_INDEX_SPACE 6
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_INDEX_MERGE_THRESHOLD 7
|
|
{STRUCT_FLD(field_name, "MERGE_THRESHOLD"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to populate the information_schema.innodb_sys_indexes table with
|
|
collected index information
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_indexes(
|
|
/*======================*/
|
|
THD* thd, /*!< in: thread */
|
|
table_id_t table_id, /*!< in: table id */
|
|
dict_index_t* index, /*!< in: populated dict_index_t
|
|
struct with index info */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_indexes");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(field_store_index_name(fields[SYS_INDEX_NAME], index->name));
|
|
|
|
OK(fields[SYS_INDEX_ID]->store(longlong(index->id), true));
|
|
|
|
OK(fields[SYS_INDEX_TABLE_ID]->store(longlong(table_id), true));
|
|
|
|
OK(fields[SYS_INDEX_TYPE]->store(index->type));
|
|
|
|
OK(fields[SYS_INDEX_NUM_FIELDS]->store(index->n_fields));
|
|
|
|
/* FIL_NULL is ULINT32_UNDEFINED */
|
|
if (index->page == FIL_NULL) {
|
|
OK(fields[SYS_INDEX_PAGE_NO]->store(-1));
|
|
} else {
|
|
OK(fields[SYS_INDEX_PAGE_NO]->store(index->page));
|
|
}
|
|
|
|
OK(fields[SYS_INDEX_SPACE]->store(index->space));
|
|
|
|
OK(fields[SYS_INDEX_MERGE_THRESHOLD]->store(index->merge_threshold));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to go through each record in SYS_INDEXES table, and fill the
|
|
information_schema.innodb_sys_indexes table with related index information
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_indexes_fill_table(
|
|
/*=======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_indexes_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
/* Start scan the SYS_INDEXES table */
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_INDEXES);
|
|
|
|
/* Process each record in the table */
|
|
while (rec) {
|
|
const char* err_msg;
|
|
table_id_t table_id;
|
|
dict_index_t index_rec;
|
|
|
|
/* Populate a dict_index_t structure with information from
|
|
a SYS_INDEXES row */
|
|
err_msg = dict_process_sys_indexes_rec(heap, rec, &index_rec,
|
|
&table_id);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_indexes(thd, table_id, &index_rec,
|
|
tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_indexes
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_indexes_init(
|
|
/*====================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_indexes_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sysindex_fields_info;
|
|
schema->fill_table = i_s_sys_indexes_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_indexes =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_INDEXES"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_INDEXES"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_indexes_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_COLUMNS **************************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_COLUMNS */
|
|
static ST_FIELD_INFO innodb_sys_columns_fields_info[] =
|
|
{
|
|
#define SYS_COLUMN_TABLE_ID 0
|
|
{STRUCT_FLD(field_name, "TABLE_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_COLUMN_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_COLUMN_POSITION 2
|
|
{STRUCT_FLD(field_name, "POS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_COLUMN_MTYPE 3
|
|
{STRUCT_FLD(field_name, "MTYPE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_COLUMN__PRTYPE 4
|
|
{STRUCT_FLD(field_name, "PRTYPE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_COLUMN_COLUMN_LEN 5
|
|
{STRUCT_FLD(field_name, "LEN"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to populate the information_schema.innodb_sys_columns with
|
|
related column information
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_columns(
|
|
/*======================*/
|
|
THD* thd, /*!< in: thread */
|
|
table_id_t table_id, /*!< in: table ID */
|
|
const char* col_name, /*!< in: column name */
|
|
dict_col_t* column, /*!< in: dict_col_t struct holding
|
|
more column information */
|
|
ulint nth_v_col, /*!< in: virtual column, its
|
|
sequence number (nth virtual col) */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_columns");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(fields[SYS_COLUMN_TABLE_ID]->store((longlong) table_id, TRUE));
|
|
|
|
OK(field_store_string(fields[SYS_COLUMN_NAME], col_name));
|
|
|
|
if (dict_col_is_virtual(column)) {
|
|
ulint pos = dict_create_v_col_pos(nth_v_col, column->ind);
|
|
OK(fields[SYS_COLUMN_POSITION]->store(pos, true));
|
|
} else {
|
|
OK(fields[SYS_COLUMN_POSITION]->store(column->ind, true));
|
|
}
|
|
|
|
OK(fields[SYS_COLUMN_MTYPE]->store(column->mtype));
|
|
|
|
OK(fields[SYS_COLUMN__PRTYPE]->store(column->prtype));
|
|
|
|
OK(fields[SYS_COLUMN_COLUMN_LEN]->store(column->len));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to fill information_schema.innodb_sys_columns with information
|
|
collected by scanning SYS_COLUMNS table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_columns_fill_table(
|
|
/*=======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
const char* col_name;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_columns_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_COLUMNS);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
dict_col_t column_rec;
|
|
table_id_t table_id;
|
|
ulint nth_v_col;
|
|
|
|
/* populate a dict_col_t structure with information from
|
|
a SYS_COLUMNS row */
|
|
err_msg = dict_process_sys_columns_rec(heap, rec, &column_rec,
|
|
&table_id, &col_name,
|
|
&nth_v_col);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_columns(thd, table_id, col_name,
|
|
&column_rec, nth_v_col,
|
|
tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_columns
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_columns_init(
|
|
/*====================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_columns_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_columns_fields_info;
|
|
schema->fill_table = i_s_sys_columns_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_columns =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_COLUMNS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_COLUMNS"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_columns_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_VIRTUAL **************************************************/
|
|
/** Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_VIRTUAL */
|
|
static ST_FIELD_INFO innodb_sys_virtual_fields_info[] =
|
|
{
|
|
#define SYS_VIRTUAL_TABLE_ID 0
|
|
{STRUCT_FLD(field_name, "TABLE_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_VIRTUAL_POS 1
|
|
{STRUCT_FLD(field_name, "POS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_VIRTUAL_BASE_POS 2
|
|
{STRUCT_FLD(field_name, "BASE_POS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/** Function to populate the information_schema.innodb_sys_virtual with
|
|
related information
|
|
param[in] thd thread
|
|
param[in] table_id table ID
|
|
param[in] pos virtual column position
|
|
param[in] base_pos base column position
|
|
param[in,out] table_to_fill fill this table
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_virtual(
|
|
THD* thd,
|
|
table_id_t table_id,
|
|
ulint pos,
|
|
ulint base_pos,
|
|
TABLE* table_to_fill)
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_virtual");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(fields[SYS_VIRTUAL_TABLE_ID]->store((longlong) table_id, TRUE));
|
|
|
|
OK(fields[SYS_VIRTUAL_POS]->store(pos));
|
|
|
|
OK(fields[SYS_VIRTUAL_BASE_POS]->store(base_pos));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/** Function to fill information_schema.innodb_sys_virtual with information
|
|
collected by scanning SYS_VIRTUAL table.
|
|
param[in] thd thread
|
|
param[in,out] tables tables to fill
|
|
param[in] item condition (not used)
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_virtual_fill_table(
|
|
THD* thd,
|
|
TABLE_LIST* tables,
|
|
Item* )
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
ulint pos;
|
|
ulint base_pos;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_virtual_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_VIRTUAL);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
table_id_t table_id;
|
|
|
|
/* populate a dict_col_t structure with information from
|
|
a SYS_VIRTUAL row */
|
|
err_msg = dict_process_sys_virtual_rec(heap, rec,
|
|
&table_id, &pos,
|
|
&base_pos);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_virtual(thd, table_id, pos, base_pos,
|
|
tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/** Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_virtual
|
|
param[in,out] p table schema object
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_virtual_init(
|
|
void* p)
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_virtual_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_virtual_fields_info;
|
|
schema->fill_table = i_s_sys_virtual_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
struct st_maria_plugin i_s_innodb_sys_virtual =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_VIRTUAL"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_VIRTUAL"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_virtual_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_BETA),
|
|
};
|
|
/** SYS_FIELDS ***************************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_FIELDS */
|
|
static ST_FIELD_INFO innodb_sys_fields_fields_info[] =
|
|
{
|
|
#define SYS_FIELD_INDEX_ID 0
|
|
{STRUCT_FLD(field_name, "INDEX_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FIELD_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FIELD_POS 2
|
|
{STRUCT_FLD(field_name, "POS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to fill information_schema.innodb_sys_fields with information
|
|
collected by scanning SYS_FIELDS table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_fields(
|
|
/*=====================*/
|
|
THD* thd, /*!< in: thread */
|
|
index_id_t index_id, /*!< in: index id for the field */
|
|
dict_field_t* field, /*!< in: table */
|
|
ulint pos, /*!< in: Field position */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_fields");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(fields[SYS_FIELD_INDEX_ID]->store((longlong) index_id, TRUE));
|
|
|
|
OK(field_store_string(fields[SYS_FIELD_NAME], field->name));
|
|
|
|
OK(fields[SYS_FIELD_POS]->store(static_cast<double>(pos)));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to go through each record in SYS_FIELDS table, and fill the
|
|
information_schema.innodb_sys_fields table with related index field
|
|
information
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_fields_fill_table(
|
|
/*======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
index_id_t last_id;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_fields_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
/* will save last index id so that we know whether we move to
|
|
the next index. This is used to calculate prefix length */
|
|
last_id = 0;
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_FIELDS);
|
|
|
|
while (rec) {
|
|
ulint pos;
|
|
const char* err_msg;
|
|
index_id_t index_id;
|
|
dict_field_t field_rec;
|
|
|
|
/* Populate a dict_field_t structure with information from
|
|
a SYS_FIELDS row */
|
|
err_msg = dict_process_sys_fields_rec(heap, rec, &field_rec,
|
|
&pos, &index_id, last_id);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_fields(thd, index_id, &field_rec,
|
|
pos, tables->table);
|
|
last_id = index_id;
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_fields
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_fields_init(
|
|
/*===================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_field_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_fields_fields_info;
|
|
schema->fill_table = i_s_sys_fields_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_fields =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_FIELDS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_FIELDS"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_fields_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_FOREIGN ********************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_FOREIGN */
|
|
static ST_FIELD_INFO innodb_sys_foreign_fields_info[] =
|
|
{
|
|
#define SYS_FOREIGN_ID 0
|
|
{STRUCT_FLD(field_name, "ID"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FOREIGN_FOR_NAME 1
|
|
{STRUCT_FLD(field_name, "FOR_NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FOREIGN_REF_NAME 2
|
|
{STRUCT_FLD(field_name, "REF_NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FOREIGN_NUM_COL 3
|
|
{STRUCT_FLD(field_name, "N_COLS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FOREIGN_TYPE 4
|
|
{STRUCT_FLD(field_name, "TYPE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to fill information_schema.innodb_sys_foreign with information
|
|
collected by scanning SYS_FOREIGN table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_foreign(
|
|
/*======================*/
|
|
THD* thd, /*!< in: thread */
|
|
dict_foreign_t* foreign, /*!< in: table */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_foreign");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(field_store_string(fields[SYS_FOREIGN_ID], foreign->id));
|
|
|
|
OK(field_store_string(fields[SYS_FOREIGN_FOR_NAME],
|
|
foreign->foreign_table_name));
|
|
|
|
OK(field_store_string(fields[SYS_FOREIGN_REF_NAME],
|
|
foreign->referenced_table_name));
|
|
|
|
OK(fields[SYS_FOREIGN_NUM_COL]->store(foreign->n_fields));
|
|
|
|
OK(fields[SYS_FOREIGN_TYPE]->store(foreign->type));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.innodb_sys_foreign table. Loop
|
|
through each record in SYS_FOREIGN, and extract the foreign key
|
|
information.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_foreign_fill_table(
|
|
/*=======================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_foreign_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_FOREIGN);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
dict_foreign_t foreign_rec;
|
|
|
|
/* Populate a dict_foreign_t structure with information from
|
|
a SYS_FOREIGN row */
|
|
err_msg = dict_process_sys_foreign_rec(heap, rec, &foreign_rec);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_foreign(thd, &foreign_rec,
|
|
tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mtr_start(&mtr);
|
|
mutex_enter(&dict_sys->mutex);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_foreign
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_foreign_init(
|
|
/*====================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_foreign_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_foreign_fields_info;
|
|
schema->fill_table = i_s_sys_foreign_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_foreign =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_FOREIGN"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_FOREIGN"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_foreign_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_FOREIGN_COLS ********************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_FOREIGN_COLS */
|
|
static ST_FIELD_INFO innodb_sys_foreign_cols_fields_info[] =
|
|
{
|
|
#define SYS_FOREIGN_COL_ID 0
|
|
{STRUCT_FLD(field_name, "ID"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FOREIGN_COL_FOR_NAME 1
|
|
{STRUCT_FLD(field_name, "FOR_COL_NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FOREIGN_COL_REF_NAME 2
|
|
{STRUCT_FLD(field_name, "REF_COL_NAME"),
|
|
STRUCT_FLD(field_length, NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_FOREIGN_COL_POS 3
|
|
{STRUCT_FLD(field_name, "POS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to fill information_schema.innodb_sys_foreign_cols with information
|
|
collected by scanning SYS_FOREIGN_COLS table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_foreign_cols(
|
|
/*==========================*/
|
|
THD* thd, /*!< in: thread */
|
|
const char* name, /*!< in: foreign key constraint name */
|
|
const char* for_col_name, /*!< in: referencing column name*/
|
|
const char* ref_col_name, /*!< in: referenced column
|
|
name */
|
|
ulint pos, /*!< in: column position */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_foreign_cols");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(field_store_string(fields[SYS_FOREIGN_COL_ID], name));
|
|
|
|
OK(field_store_string(fields[SYS_FOREIGN_COL_FOR_NAME], for_col_name));
|
|
|
|
OK(field_store_string(fields[SYS_FOREIGN_COL_REF_NAME], ref_col_name));
|
|
|
|
OK(fields[SYS_FOREIGN_COL_POS]->store(pos, true));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.innodb_sys_foreign_cols table. Loop
|
|
through each record in SYS_FOREIGN_COLS, and extract the foreign key column
|
|
information and fill the INFORMATION_SCHEMA.innodb_sys_foreign_cols table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_foreign_cols_fill_table(
|
|
/*============================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_foreign_cols_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_FOREIGN_COLS);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
const char* name;
|
|
const char* for_col_name;
|
|
const char* ref_col_name;
|
|
ulint pos;
|
|
|
|
/* Extract necessary information from a SYS_FOREIGN_COLS row */
|
|
err_msg = dict_process_sys_foreign_col_rec(
|
|
heap, rec, &name, &for_col_name, &ref_col_name, &pos);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_foreign_cols(
|
|
thd, name, for_col_name, ref_col_name, pos,
|
|
tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.innodb_sys_foreign_cols
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_foreign_cols_init(
|
|
/*========================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_foreign_cols_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_foreign_cols_fields_info;
|
|
schema->fill_table = i_s_sys_foreign_cols_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_foreign_cols =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_FOREIGN_COLS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_FOREIGN_COLS"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_foreign_cols_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_TABLESPACES ********************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_TABLESPACES */
|
|
static ST_FIELD_INFO innodb_sys_tablespaces_fields_info[] =
|
|
{
|
|
#define SYS_TABLESPACES_SPACE 0
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, MAX_FULL_NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_FLAGS 2
|
|
{STRUCT_FLD(field_name, "FLAG"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_FILE_FORMAT 3
|
|
{STRUCT_FLD(field_name, "FILE_FORMAT"),
|
|
STRUCT_FLD(field_length, 10),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_ROW_FORMAT 4
|
|
{STRUCT_FLD(field_name, "ROW_FORMAT"),
|
|
STRUCT_FLD(field_length, 22),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_PAGE_SIZE 5
|
|
{STRUCT_FLD(field_name, "PAGE_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_ZIP_PAGE_SIZE 6
|
|
{STRUCT_FLD(field_name, "ZIP_PAGE_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_SPACE_TYPE 7
|
|
{STRUCT_FLD(field_name, "SPACE_TYPE"),
|
|
STRUCT_FLD(field_length, 10),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_FS_BLOCK_SIZE 8
|
|
{STRUCT_FLD(field_name, "FS_BLOCK_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_FILE_SIZE 9
|
|
{STRUCT_FLD(field_name, "FILE_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_TABLESPACES_ALLOC_SIZE 10
|
|
{STRUCT_FLD(field_name, "ALLOCATED_SIZE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to fill INFORMATION_SCHEMA.INNODB_SYS_TABLESPACES with information
|
|
collected by scanning SYS_TABLESPACESS table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_tablespaces(
|
|
/*==========================*/
|
|
THD* thd, /*!< in: thread */
|
|
ulint space, /*!< in: space ID */
|
|
const char* name, /*!< in: tablespace name */
|
|
ulint flags, /*!< in: tablespace flags */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
ulint atomic_blobs = FSP_FLAGS_HAS_ATOMIC_BLOBS(flags);
|
|
bool is_compressed = FSP_FLAGS_GET_ZIP_SSIZE(flags);
|
|
const char* file_format;
|
|
const char* row_format;
|
|
const page_size_t page_size(flags);
|
|
const char* space_type;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_tablespaces");
|
|
|
|
file_format = trx_sys_file_format_id_to_name(atomic_blobs);
|
|
if (is_system_tablespace(space)) {
|
|
row_format = "Compact or Redundant";
|
|
} else if (is_compressed) {
|
|
row_format = "Compressed";
|
|
} else if (atomic_blobs) {
|
|
row_format = "Dynamic";
|
|
} else {
|
|
row_format = "Compact or Redundant";
|
|
}
|
|
|
|
if (is_system_tablespace(space)) {
|
|
space_type = "System";
|
|
} else {
|
|
space_type = "Single";
|
|
}
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(fields[SYS_TABLESPACES_SPACE]->store(space, true));
|
|
|
|
OK(field_store_string(fields[SYS_TABLESPACES_NAME], name));
|
|
|
|
OK(fields[SYS_TABLESPACES_FLAGS]->store(flags, true));
|
|
|
|
OK(field_store_string(fields[SYS_TABLESPACES_FILE_FORMAT],
|
|
file_format));
|
|
|
|
OK(field_store_string(fields[SYS_TABLESPACES_ROW_FORMAT], row_format));
|
|
|
|
OK(fields[SYS_TABLESPACES_PAGE_SIZE]->store(
|
|
univ_page_size.physical(), true));
|
|
|
|
OK(fields[SYS_TABLESPACES_ZIP_PAGE_SIZE]->store(
|
|
page_size.is_compressed()
|
|
? page_size.physical()
|
|
: 0, true));
|
|
|
|
OK(field_store_string(fields[SYS_TABLESPACES_SPACE_TYPE],
|
|
space_type));
|
|
|
|
char* filepath = NULL;
|
|
if (FSP_FLAGS_HAS_DATA_DIR(flags)) {
|
|
mutex_enter(&dict_sys->mutex);
|
|
filepath = dict_get_first_path(space);
|
|
mutex_exit(&dict_sys->mutex);
|
|
}
|
|
|
|
if (filepath == NULL) {
|
|
filepath = fil_make_filepath(NULL, name, IBD, false);
|
|
}
|
|
|
|
os_file_stat_t stat;
|
|
os_file_size_t file;
|
|
|
|
memset(&file, 0xff, sizeof(file));
|
|
memset(&stat, 0x0, sizeof(stat));
|
|
|
|
if (filepath != NULL) {
|
|
|
|
file = os_file_get_size(filepath);
|
|
|
|
/* Get the file system (or Volume) block size. */
|
|
dberr_t err = os_file_get_status(filepath, &stat, false, false);
|
|
|
|
switch(err) {
|
|
case DB_FAIL:
|
|
ib::warn()
|
|
<< "File '" << filepath << "', failed to get "
|
|
<< "stats";
|
|
break;
|
|
|
|
case DB_SUCCESS:
|
|
case DB_NOT_FOUND:
|
|
break;
|
|
|
|
default:
|
|
ib::error()
|
|
<< "File '" << filepath << "' "
|
|
<< ut_strerr(err);
|
|
break;
|
|
}
|
|
|
|
ut_free(filepath);
|
|
}
|
|
|
|
if (file.m_total_size == static_cast<os_offset_t>(~0)) {
|
|
stat.block_size = 0;
|
|
file.m_total_size = 0;
|
|
file.m_alloc_size = 0;
|
|
}
|
|
|
|
OK(fields[SYS_TABLESPACES_FS_BLOCK_SIZE]->store(stat.block_size, true));
|
|
|
|
OK(fields[SYS_TABLESPACES_FILE_SIZE]->store(file.m_total_size, true));
|
|
|
|
OK(fields[SYS_TABLESPACES_ALLOC_SIZE]->store(file.m_alloc_size, true));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.INNODB_SYS_TABLESPACES table.
|
|
Loop through each record in SYS_TABLESPACES, and extract the column
|
|
information and fill the INFORMATION_SCHEMA.INNODB_SYS_TABLESPACES table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_tablespaces_fill_table(
|
|
/*===========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_tablespaces_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
for (rec = dict_startscan_system(&pcur, &mtr, SYS_TABLESPACES);
|
|
rec != NULL;
|
|
rec = dict_getnext_system(&pcur, &mtr)) {
|
|
|
|
const char* err_msg;
|
|
ulint space;
|
|
const char* name;
|
|
ulint flags;
|
|
|
|
/* Extract necessary information from a SYS_TABLESPACES row */
|
|
err_msg = dict_process_sys_tablespaces(
|
|
heap, rec, &space, &name, &flags);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_tablespaces(
|
|
thd, space, name, flags,
|
|
tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_SYS_TABLESPACES
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_tablespaces_init(
|
|
/*========================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_tablespaces_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_tablespaces_fields_info;
|
|
schema->fill_table = i_s_sys_tablespaces_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_tablespaces =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_TABLESPACES"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_TABLESPACES"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_tablespaces_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_DATAFILES ************************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_DATAFILES */
|
|
static ST_FIELD_INFO innodb_sys_datafiles_fields_info[] =
|
|
{
|
|
#define SYS_DATAFILES_SPACE 0
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define SYS_DATAFILES_PATH 1
|
|
{STRUCT_FLD(field_name, "PATH"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to fill INFORMATION_SCHEMA.INNODB_SYS_DATAFILES with information
|
|
collected by scanning SYS_DATAFILESS table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_sys_datafiles(
|
|
/*========================*/
|
|
THD* thd, /*!< in: thread */
|
|
ulint space, /*!< in: space ID */
|
|
const char* path, /*!< in: absolute path */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_sys_datafiles");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
OK(field_store_ulint(fields[SYS_DATAFILES_SPACE], space));
|
|
|
|
OK(field_store_string(fields[SYS_DATAFILES_PATH], path));
|
|
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.INNODB_SYS_DATAFILES table.
|
|
Loop through each record in SYS_DATAFILES, and extract the column
|
|
information and fill the INFORMATION_SCHEMA.INNODB_SYS_DATAFILES table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_sys_datafiles_fill_table(
|
|
/*=========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
|
|
DBUG_ENTER("i_s_sys_datafiles_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_DATAFILES);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
ulint space;
|
|
const char* path;
|
|
|
|
/* Extract necessary information from a SYS_DATAFILES row */
|
|
err_msg = dict_process_sys_datafiles(
|
|
heap, rec, &space, &path);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_sys_datafiles(
|
|
thd, space, path, tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_SYS_DATAFILES
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_datafiles_init(
|
|
/*======================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_datafiles_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_datafiles_fields_info;
|
|
schema->fill_table = i_s_sys_datafiles_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_datafiles =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_DATAFILES"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_DATAFILES"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_datafiles_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** TABLESPACES_ENCRYPTION ********************************************/
|
|
/* Fields of the table INFORMATION_SCHEMA.INNODB_TABLESPACES_ENCRYPTION */
|
|
static ST_FIELD_INFO innodb_tablespaces_encryption_fields_info[] =
|
|
{
|
|
#define TABLESPACES_ENCRYPTION_SPACE 0
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, MAX_FULL_NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_ENCRYPTION_SCHEME 2
|
|
{STRUCT_FLD(field_name, "ENCRYPTION_SCHEME"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_KEYSERVER_REQUESTS 3
|
|
{STRUCT_FLD(field_name, "KEYSERVER_REQUESTS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_MIN_KEY_VERSION 4
|
|
{STRUCT_FLD(field_name, "MIN_KEY_VERSION"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_CURRENT_KEY_VERSION 5
|
|
{STRUCT_FLD(field_name, "CURRENT_KEY_VERSION"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_KEY_ROTATION_PAGE_NUMBER 6
|
|
{STRUCT_FLD(field_name, "KEY_ROTATION_PAGE_NUMBER"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED | MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_KEY_ROTATION_MAX_PAGE_NUMBER 7
|
|
{STRUCT_FLD(field_name, "KEY_ROTATION_MAX_PAGE_NUMBER"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED | MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_ENCRYPTION_CURRENT_KEY_ID 8
|
|
{STRUCT_FLD(field_name, "CURRENT_KEY_ID"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to fill INFORMATION_SCHEMA.INNODB_TABLESPACES_ENCRYPTION
|
|
with information collected by scanning SYS_TABLESPACES table and then use
|
|
fil_space()
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_tablespaces_encryption(
|
|
/*==========================*/
|
|
THD* thd, /*!< in: thread */
|
|
ulint space, /*!< in: space ID */
|
|
const char* name, /*!< in: tablespace name */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
struct fil_space_crypt_status_t status;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_tablespaces_encryption");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
fil_space_crypt_get_status(space, &status);
|
|
OK(fields[TABLESPACES_ENCRYPTION_SPACE]->store(space));
|
|
|
|
OK(field_store_string(fields[TABLESPACES_ENCRYPTION_NAME],
|
|
name));
|
|
|
|
OK(fields[TABLESPACES_ENCRYPTION_ENCRYPTION_SCHEME]->store(
|
|
status.scheme));
|
|
OK(fields[TABLESPACES_ENCRYPTION_KEYSERVER_REQUESTS]->store(
|
|
status.keyserver_requests));
|
|
OK(fields[TABLESPACES_ENCRYPTION_MIN_KEY_VERSION]->store(
|
|
status.min_key_version));
|
|
OK(fields[TABLESPACES_ENCRYPTION_CURRENT_KEY_VERSION]->store(
|
|
status.current_key_version));
|
|
OK(fields[TABLESPACES_ENCRYPTION_CURRENT_KEY_ID]->store(
|
|
status.key_id));
|
|
if (status.rotating) {
|
|
fields[TABLESPACES_ENCRYPTION_KEY_ROTATION_PAGE_NUMBER]->set_notnull();
|
|
OK(fields[TABLESPACES_ENCRYPTION_KEY_ROTATION_PAGE_NUMBER]->store(
|
|
status.rotate_next_page_number));
|
|
fields[TABLESPACES_ENCRYPTION_KEY_ROTATION_MAX_PAGE_NUMBER]->set_notnull();
|
|
OK(fields[TABLESPACES_ENCRYPTION_KEY_ROTATION_MAX_PAGE_NUMBER]->store(
|
|
status.rotate_max_page_number));
|
|
} else {
|
|
fields[TABLESPACES_ENCRYPTION_KEY_ROTATION_PAGE_NUMBER]
|
|
->set_null();
|
|
fields[TABLESPACES_ENCRYPTION_KEY_ROTATION_MAX_PAGE_NUMBER]
|
|
->set_null();
|
|
}
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.INNODB_TABLESPACES_ENCRYPTION table.
|
|
Loop through each record in TABLESPACES_ENCRYPTION, and extract the column
|
|
information and fill the INFORMATION_SCHEMA.INNODB_TABLESPACES_ENCRYPTION table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_tablespaces_encryption_fill_table(
|
|
/*===========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
bool found_space_0 = false;
|
|
|
|
DBUG_ENTER("i_s_tablespaces_encryption_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, SUPER_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_TABLESPACES);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
ulint space;
|
|
const char* name;
|
|
ulint flags;
|
|
|
|
/* Extract necessary information from a SYS_TABLESPACES row */
|
|
err_msg = dict_process_sys_tablespaces(
|
|
heap, rec, &space, &name, &flags);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (space == 0) {
|
|
found_space_0 = true;
|
|
}
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_tablespaces_encryption(
|
|
thd, space, name, tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
if (found_space_0 == false) {
|
|
/* space 0 does for what ever unknown reason not show up
|
|
* in iteration above, add it manually */
|
|
ulint space = 0;
|
|
const char* name = NULL;
|
|
i_s_dict_fill_tablespaces_encryption(
|
|
thd, space, name, tables->table);
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_TABLESPACES_ENCRYPTION
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_tablespaces_encryption_init(
|
|
/*========================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_tablespaces_encryption_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_tablespaces_encryption_fields_info;
|
|
schema->fill_table = i_s_tablespaces_encryption_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_tablespaces_encryption =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_TABLESPACES_ENCRYPTION"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, "Google Inc"),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB TABLESPACES_ENCRYPTION"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_BSD),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_tablespaces_encryption_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE)
|
|
};
|
|
|
|
/** TABLESPACES_SCRUBBING ********************************************/
|
|
/* Fields of the table INFORMATION_SCHEMA.INNODB_TABLESPACES_SCRUBBING */
|
|
static ST_FIELD_INFO innodb_tablespaces_scrubbing_fields_info[] =
|
|
{
|
|
#define TABLESPACES_SCRUBBING_SPACE 0
|
|
{STRUCT_FLD(field_name, "SPACE"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_SCRUBBING_NAME 1
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, MAX_FULL_NAME_LEN + 1),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_SCRUBBING_COMPRESSED 2
|
|
{STRUCT_FLD(field_name, "COMPRESSED"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_SCRUBBING_LAST_SCRUB_COMPLETED 3
|
|
{STRUCT_FLD(field_name, "LAST_SCRUB_COMPLETED"),
|
|
STRUCT_FLD(field_length, 0),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_DATETIME),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_SCRUBBING_CURRENT_SCRUB_STARTED 4
|
|
{STRUCT_FLD(field_name, "CURRENT_SCRUB_STARTED"),
|
|
STRUCT_FLD(field_length, 0),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_DATETIME),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_SCRUBBING_CURRENT_SCRUB_ACTIVE_THREADS 5
|
|
{STRUCT_FLD(field_name, "CURRENT_SCRUB_ACTIVE_THREADS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED | MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_SCRUBBING_CURRENT_SCRUB_PAGE_NUMBER 6
|
|
{STRUCT_FLD(field_name, "CURRENT_SCRUB_PAGE_NUMBER"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
#define TABLESPACES_SCRUBBING_CURRENT_SCRUB_MAX_PAGE_NUMBER 7
|
|
{STRUCT_FLD(field_name, "CURRENT_SCRUB_MAX_PAGE_NUMBER"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/**********************************************************************//**
|
|
Function to fill INFORMATION_SCHEMA.INNODB_TABLESPACES_SCRUBBING
|
|
with information collected by scanning SYS_TABLESPACES table and then use
|
|
fil_space()
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_dict_fill_tablespaces_scrubbing(
|
|
/*==========================*/
|
|
THD* thd, /*!< in: thread */
|
|
ulint space, /*!< in: space ID */
|
|
const char* name, /*!< in: tablespace name */
|
|
TABLE* table_to_fill) /*!< in/out: fill this table */
|
|
{
|
|
Field** fields;
|
|
struct fil_space_scrub_status_t status;
|
|
|
|
DBUG_ENTER("i_s_dict_fill_tablespaces_scrubbing");
|
|
|
|
fields = table_to_fill->field;
|
|
|
|
fil_space_get_scrub_status(space, &status);
|
|
OK(fields[TABLESPACES_SCRUBBING_SPACE]->store(space));
|
|
|
|
OK(field_store_string(fields[TABLESPACES_SCRUBBING_NAME],
|
|
name));
|
|
|
|
OK(fields[TABLESPACES_SCRUBBING_COMPRESSED]->store(
|
|
status.compressed ? 1 : 0));
|
|
|
|
if (status.last_scrub_completed == 0) {
|
|
fields[TABLESPACES_SCRUBBING_LAST_SCRUB_COMPLETED]->set_null();
|
|
} else {
|
|
fields[TABLESPACES_SCRUBBING_LAST_SCRUB_COMPLETED]
|
|
->set_notnull();
|
|
OK(field_store_time_t(
|
|
fields[TABLESPACES_SCRUBBING_LAST_SCRUB_COMPLETED],
|
|
status.last_scrub_completed));
|
|
}
|
|
|
|
int field_numbers[] = {
|
|
TABLESPACES_SCRUBBING_CURRENT_SCRUB_STARTED,
|
|
TABLESPACES_SCRUBBING_CURRENT_SCRUB_ACTIVE_THREADS,
|
|
TABLESPACES_SCRUBBING_CURRENT_SCRUB_PAGE_NUMBER,
|
|
TABLESPACES_SCRUBBING_CURRENT_SCRUB_MAX_PAGE_NUMBER };
|
|
if (status.scrubbing) {
|
|
for (uint i = 0; i < array_elements(field_numbers); i++) {
|
|
fields[field_numbers[i]]->set_notnull();
|
|
}
|
|
|
|
OK(field_store_time_t(
|
|
fields[TABLESPACES_SCRUBBING_CURRENT_SCRUB_STARTED],
|
|
status.current_scrub_started));
|
|
OK(fields[TABLESPACES_SCRUBBING_CURRENT_SCRUB_ACTIVE_THREADS]
|
|
->store(status.current_scrub_active_threads));
|
|
OK(fields[TABLESPACES_SCRUBBING_CURRENT_SCRUB_PAGE_NUMBER]
|
|
->store(status.current_scrub_page_number));
|
|
OK(fields[TABLESPACES_SCRUBBING_CURRENT_SCRUB_MAX_PAGE_NUMBER]
|
|
->store(status.current_scrub_max_page_number));
|
|
} else {
|
|
for (uint i = 0; i < array_elements(field_numbers); i++) {
|
|
fields[field_numbers[i]]->set_null();
|
|
}
|
|
}
|
|
OK(schema_table_store_record(thd, table_to_fill));
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.INNODB_TABLESPACES_SCRUBBING table.
|
|
Loop through each record in TABLESPACES_SCRUBBING, and extract the column
|
|
information and fill the INFORMATION_SCHEMA.INNODB_TABLESPACES_SCRUBBING table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_tablespaces_scrubbing_fill_table(
|
|
/*===========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
btr_pcur_t pcur;
|
|
const rec_t* rec;
|
|
mem_heap_t* heap;
|
|
mtr_t mtr;
|
|
bool found_space_0 = false;
|
|
|
|
DBUG_ENTER("i_s_tablespaces_scrubbing_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without SUPER_ACL privilege */
|
|
if (check_global_access(thd, SUPER_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
heap = mem_heap_create(1000);
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
|
|
rec = dict_startscan_system(&pcur, &mtr, SYS_TABLESPACES);
|
|
|
|
while (rec) {
|
|
const char* err_msg;
|
|
ulint space;
|
|
const char* name;
|
|
ulint flags;
|
|
|
|
/* Extract necessary information from a SYS_TABLESPACES row */
|
|
err_msg = dict_process_sys_tablespaces(
|
|
heap, rec, &space, &name, &flags);
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
if (space == 0) {
|
|
found_space_0 = true;
|
|
}
|
|
|
|
if (!err_msg) {
|
|
i_s_dict_fill_tablespaces_scrubbing(
|
|
thd, space, name, tables->table);
|
|
} else {
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
ER_CANT_FIND_SYSTEM_REC, "%s",
|
|
err_msg);
|
|
}
|
|
|
|
mem_heap_empty(heap);
|
|
|
|
/* Get the next record */
|
|
mutex_enter(&dict_sys->mutex);
|
|
mtr_start(&mtr);
|
|
rec = dict_getnext_system(&pcur, &mtr);
|
|
}
|
|
|
|
mtr_commit(&mtr);
|
|
mutex_exit(&dict_sys->mutex);
|
|
mem_heap_free(heap);
|
|
|
|
if (found_space_0 == false) {
|
|
/* space 0 does for what ever unknown reason not show up
|
|
* in iteration above, add it manually */
|
|
ulint space = 0;
|
|
const char* name = NULL;
|
|
i_s_dict_fill_tablespaces_scrubbing(
|
|
thd, space, name, tables->table);
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_TABLESPACES_SCRUBBING
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_tablespaces_scrubbing_init(
|
|
/*========================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_tablespaces_scrubbing_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_tablespaces_scrubbing_fields_info;
|
|
schema->fill_table = i_s_tablespaces_scrubbing_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_tablespaces_scrubbing =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_TABLESPACES_SCRUBBING"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, "Google Inc"),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB TABLESPACES_SCRUBBING"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_BSD),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_tablespaces_scrubbing_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE)
|
|
};
|
|
|
|
/** INNODB_MUTEXES *********************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_MUTEXES */
|
|
static ST_FIELD_INFO innodb_mutexes_fields_info[] =
|
|
{
|
|
#define MUTEXES_NAME 0
|
|
{STRUCT_FLD(field_name, "NAME"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
#define MUTEXES_CREATE_FILE 1
|
|
{STRUCT_FLD(field_name, "CREATE_FILE"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, 0),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
#define MUTEXES_CREATE_LINE 2
|
|
{STRUCT_FLD(field_name, "CREATE_LINE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
#define MUTEXES_OS_WAITS 3
|
|
{STRUCT_FLD(field_name, "OS_WAITS"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
/*******************************************************************//**
|
|
Function to populate INFORMATION_SCHEMA.INNODB_MUTEXES table.
|
|
Loop through each record in mutex and rw_lock lists, and extract the column
|
|
information and fill the INFORMATION_SCHEMA.INNODB_MUTEXES table.
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
i_s_innodb_mutexes_fill_table(
|
|
/*==========================*/
|
|
THD* thd, /*!< in: thread */
|
|
TABLE_LIST* tables, /*!< in/out: tables to fill */
|
|
Item* ) /*!< in: condition (not used) */
|
|
{
|
|
rw_lock_t* lock;
|
|
ulint block_lock_oswait_count = 0;
|
|
rw_lock_t* block_lock = NULL;
|
|
Field** fields = tables->table->field;
|
|
|
|
DBUG_ENTER("i_s_innodb_mutexes_fill_table");
|
|
RETURN_IF_INNODB_NOT_STARTED(tables->schema_table_name);
|
|
|
|
/* deny access to user without PROCESS_ACL privilege */
|
|
if (check_global_access(thd, PROCESS_ACL)) {
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
// mutex_enter(&mutex_list_mutex);
|
|
|
|
#ifdef JAN_TODO_FIXME
|
|
ib_mutex_t* mutex;
|
|
ulint block_mutex_oswait_count = 0;
|
|
ib_mutex_t* block_mutex = NULL;
|
|
for (mutex = UT_LIST_GET_FIRST(os_mutex_list); mutex != NULL;
|
|
mutex = UT_LIST_GET_NEXT(list, mutex)) {
|
|
if (mutex->count_os_wait == 0) {
|
|
continue;
|
|
}
|
|
|
|
if (buf_pool_is_block_mutex(mutex)) {
|
|
block_mutex = mutex;
|
|
block_mutex_oswait_count += mutex->count_os_wait;
|
|
continue;
|
|
}
|
|
|
|
OK(field_store_string(fields[MUTEXES_NAME], mutex->cmutex_name));
|
|
OK(field_store_string(fields[MUTEXES_CREATE_FILE], innobase_basename(mutex->cfile_name)));
|
|
OK(field_store_ulint(fields[MUTEXES_CREATE_LINE], mutex->cline));
|
|
OK(field_store_ulint(fields[MUTEXES_OS_WAITS], (longlong)mutex->count_os_wait));
|
|
OK(schema_table_store_record(thd, tables->table));
|
|
}
|
|
|
|
if (block_mutex) {
|
|
char buf1[IO_SIZE];
|
|
|
|
my_snprintf(buf1, sizeof buf1, "combined %s",
|
|
innobase_basename(block_mutex->cfile_name));
|
|
|
|
OK(field_store_string(fields[MUTEXES_NAME], block_mutex->cmutex_name));
|
|
OK(field_store_string(fields[MUTEXES_CREATE_FILE], buf1));
|
|
OK(field_store_ulint(fields[MUTEXES_CREATE_LINE], block_mutex->cline));
|
|
OK(field_store_ulint(fields[MUTEXES_OS_WAITS], (longlong)block_mutex_oswait_count));
|
|
OK(schema_table_store_record(thd, tables->table));
|
|
}
|
|
|
|
mutex_exit(&mutex_list_mutex);
|
|
#endif /* JAN_TODO_FIXME */
|
|
|
|
mutex_enter(&rw_lock_list_mutex);
|
|
|
|
for (lock = UT_LIST_GET_FIRST(rw_lock_list); lock != NULL;
|
|
lock = UT_LIST_GET_NEXT(list, lock)) {
|
|
if (lock->count_os_wait == 0) {
|
|
continue;
|
|
}
|
|
|
|
if (buf_pool_is_block_lock(lock)) {
|
|
block_lock = lock;
|
|
block_lock_oswait_count += lock->count_os_wait;
|
|
continue;
|
|
}
|
|
|
|
//OK(field_store_string(fields[MUTEXES_NAME], lock->lock_name));
|
|
OK(field_store_string(fields[MUTEXES_CREATE_FILE], innobase_basename(lock->cfile_name)));
|
|
OK(field_store_ulint(fields[MUTEXES_CREATE_LINE], lock->cline));
|
|
OK(field_store_ulint(fields[MUTEXES_OS_WAITS], (longlong)lock->count_os_wait));
|
|
OK(schema_table_store_record(thd, tables->table));
|
|
}
|
|
|
|
if (block_lock) {
|
|
char buf1[IO_SIZE];
|
|
|
|
my_snprintf(buf1, sizeof buf1, "combined %s",
|
|
innobase_basename(block_lock->cfile_name));
|
|
|
|
//OK(field_store_string(fields[MUTEXES_NAME], block_lock->lock_name));
|
|
OK(field_store_string(fields[MUTEXES_CREATE_FILE], buf1));
|
|
OK(field_store_ulint(fields[MUTEXES_CREATE_LINE], block_lock->cline));
|
|
OK(field_store_ulint(fields[MUTEXES_OS_WAITS], (longlong)block_lock_oswait_count));
|
|
OK(schema_table_store_record(thd, tables->table));
|
|
}
|
|
|
|
mutex_exit(&rw_lock_list_mutex);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_MUTEXES
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_mutexes_init(
|
|
/*================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_mutexes_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_mutexes_fields_info;
|
|
schema->fill_table = i_s_innodb_mutexes_fill_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_mutexes =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_MUTEXES"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_DATAFILES"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_mutexes_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|
|
|
|
/** SYS_SEMAPHORE_WAITS ************************************************/
|
|
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_SYS_SEMAPHORE_WAITS */
|
|
static ST_FIELD_INFO innodb_sys_semaphore_waits_fields_info[] =
|
|
{
|
|
// SYS_SEMAPHORE_WAITS_THREAD_ID 0
|
|
{STRUCT_FLD(field_name, "THREAD_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_OBJECT_NAME 1
|
|
{STRUCT_FLD(field_name, "OBJECT_NAME"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_FILE 2
|
|
{STRUCT_FLD(field_name, "FILE"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_LINE 3
|
|
{STRUCT_FLD(field_name, "LINE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_WAIT_TIME 4
|
|
{STRUCT_FLD(field_name, "WAIT_TIME"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_WAIT_OBJECT 5
|
|
{STRUCT_FLD(field_name, "WAIT_OBJECT"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_WAIT_TYPE 6
|
|
{STRUCT_FLD(field_name, "WAIT_TYPE"),
|
|
STRUCT_FLD(field_length, 16),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_HOLDER_THREAD_ID 7
|
|
{STRUCT_FLD(field_name, "HOLDER_THREAD_ID"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_HOLDER_FILE 8
|
|
{STRUCT_FLD(field_name, "HOLDER_FILE"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_HOLDER_LINE 9
|
|
{STRUCT_FLD(field_name, "HOLDER_LINE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_CREATED_FILE 10
|
|
{STRUCT_FLD(field_name, "CREATED_FILE"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_CREATED_LINE 11
|
|
{STRUCT_FLD(field_name, "CREATED_LINE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_WRITER_THREAD 12
|
|
{STRUCT_FLD(field_name, "WRITER_THREAD"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_RESERVATION_MODE 13
|
|
{STRUCT_FLD(field_name, "RESERVATION_MODE"),
|
|
STRUCT_FLD(field_length, 16),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_READERS 14
|
|
{STRUCT_FLD(field_name, "READERS"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_WAITERS_FLAG 15
|
|
{STRUCT_FLD(field_name, "WAITERS_FLAG"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_LOCK_WORD 16
|
|
{STRUCT_FLD(field_name, "LOCK_WORD"),
|
|
STRUCT_FLD(field_length, MY_INT64_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONGLONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_LAST_READER_FILE 17
|
|
{STRUCT_FLD(field_name, "LAST_READER_FILE"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_LAST_READER_LINE 18
|
|
{STRUCT_FLD(field_name, "LAST_READER_LINE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_LAST_WRITER_FILE 19
|
|
{STRUCT_FLD(field_name, "LAST_WRITER_FILE"),
|
|
STRUCT_FLD(field_length, OS_FILE_MAX_PATH),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_STRING),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_MAYBE_NULL),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_LAST_WRITER_LINE 20
|
|
{STRUCT_FLD(field_name, "LAST_WRITER_LINE"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
// SYS_SEMAPHORE_WAITS_OS_WAIT_COUNT 21
|
|
{STRUCT_FLD(field_name, "OS_WAIT_COUNT"),
|
|
STRUCT_FLD(field_length, MY_INT32_NUM_DECIMAL_DIGITS),
|
|
STRUCT_FLD(field_type, MYSQL_TYPE_LONG),
|
|
STRUCT_FLD(value, 0),
|
|
STRUCT_FLD(field_flags, MY_I_S_UNSIGNED),
|
|
STRUCT_FLD(old_name, ""),
|
|
STRUCT_FLD(open_method, SKIP_OPEN_TABLE)},
|
|
|
|
END_OF_ST_FIELD_INFO
|
|
};
|
|
|
|
|
|
|
|
/*******************************************************************//**
|
|
Bind the dynamic table INFORMATION_SCHEMA.INNODB_SYS_SEMAPHORE_WAITS
|
|
@return 0 on success */
|
|
static
|
|
int
|
|
innodb_sys_semaphore_waits_init(
|
|
/*============================*/
|
|
void* p) /*!< in/out: table schema object */
|
|
{
|
|
ST_SCHEMA_TABLE* schema;
|
|
|
|
DBUG_ENTER("innodb_sys_semaphore_waits_init");
|
|
|
|
schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innodb_sys_semaphore_waits_fields_info;
|
|
schema->fill_table = sync_arr_fill_sys_semphore_waits_table;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
UNIV_INTERN struct st_maria_plugin i_s_innodb_sys_semaphore_waits =
|
|
{
|
|
/* the plugin type (a MYSQL_XXX_PLUGIN value) */
|
|
/* int */
|
|
STRUCT_FLD(type, MYSQL_INFORMATION_SCHEMA_PLUGIN),
|
|
|
|
/* pointer to type-specific plugin descriptor */
|
|
/* void* */
|
|
STRUCT_FLD(info, &i_s_info),
|
|
|
|
/* plugin name */
|
|
/* const char* */
|
|
STRUCT_FLD(name, "INNODB_SYS_SEMAPHORE_WAITS"),
|
|
|
|
/* plugin author (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(author, maria_plugin_author),
|
|
|
|
/* general descriptive text (for SHOW PLUGINS) */
|
|
/* const char* */
|
|
STRUCT_FLD(descr, "InnoDB SYS_SEMAPHORE_WAITS"),
|
|
|
|
/* the plugin license (PLUGIN_LICENSE_XXX) */
|
|
/* int */
|
|
STRUCT_FLD(license, PLUGIN_LICENSE_GPL),
|
|
|
|
/* the function to invoke when plugin is loaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(init, innodb_sys_semaphore_waits_init),
|
|
|
|
/* the function to invoke when plugin is unloaded */
|
|
/* int (*)(void*); */
|
|
STRUCT_FLD(deinit, i_s_common_deinit),
|
|
|
|
/* plugin version (for SHOW PLUGINS) */
|
|
/* unsigned int */
|
|
STRUCT_FLD(version, INNODB_VERSION_SHORT),
|
|
|
|
/* struct st_mysql_show_var* */
|
|
STRUCT_FLD(status_vars, NULL),
|
|
|
|
/* struct st_mysql_sys_var** */
|
|
STRUCT_FLD(system_vars, NULL),
|
|
|
|
/* Maria extension */
|
|
STRUCT_FLD(version_info, INNODB_VERSION_STR),
|
|
STRUCT_FLD(maturity, MariaDB_PLUGIN_MATURITY_STABLE),
|
|
};
|