2007-09-20 11:23:44 +00:00
|
|
|
/******************************************************
|
|
|
|
INFORMATION SCHEMA innodb_trx, innodb_locks and
|
|
|
|
innodb_lock_waits tables fetch code.
|
|
|
|
|
|
|
|
The code below fetches information needed to fill those
|
|
|
|
3 dynamic tables and uploads it into a "transactions
|
|
|
|
table cache" for later retrieval.
|
|
|
|
|
|
|
|
(c) 2007 Innobase Oy
|
|
|
|
|
|
|
|
Created July 17, 2007 Vasil Dimov
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
#include <mysql/plugin.h>
|
|
|
|
|
2007-11-07 14:14:33 +00:00
|
|
|
#include "mysql_addons.h"
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
#include "univ.i"
|
2007-10-29 13:03:53 +00:00
|
|
|
#include "buf0buf.h"
|
|
|
|
#include "dict0dict.h"
|
2007-09-24 12:25:06 +00:00
|
|
|
#include "ha0storage.h"
|
2007-09-20 11:23:44 +00:00
|
|
|
#include "hash0hash.h"
|
|
|
|
#include "lock0iter.h"
|
|
|
|
#include "lock0lock.h"
|
|
|
|
#include "mem0mem.h"
|
2007-10-29 13:03:53 +00:00
|
|
|
#include "page0page.h"
|
|
|
|
#include "rem0rec.h"
|
|
|
|
#include "row0row.h"
|
2007-09-20 11:23:44 +00:00
|
|
|
#include "srv0srv.h"
|
|
|
|
#include "sync0rw.h"
|
|
|
|
#include "sync0sync.h"
|
|
|
|
#include "sync0types.h"
|
|
|
|
#include "trx0i_s.h"
|
|
|
|
#include "trx0sys.h"
|
|
|
|
#include "trx0trx.h"
|
|
|
|
#include "ut0mem.h"
|
|
|
|
#include "ut0ut.h"
|
|
|
|
|
|
|
|
#define TABLE_CACHE_INITIAL_ROWSNUM 1024
|
|
|
|
|
|
|
|
/* Table cache's rows are stored in a set of chunks. When a new row is
|
|
|
|
added a new chunk is allocated if necessary. MEM_CHUNKS_IN_TABLE_CACHE
|
|
|
|
specifies the maximum number of chunks.
|
|
|
|
Assuming that the first one is 1024 rows (TABLE_CACHE_INITIAL_ROWSNUM)
|
|
|
|
and each subsequent is N/2 where N is the number of rows we have
|
|
|
|
allocated till now, then 39th chunk would have 1677416425 number of rows
|
|
|
|
and all chunks would have 3354832851 number of rows. */
|
|
|
|
#define MEM_CHUNKS_IN_TABLE_CACHE 39
|
|
|
|
|
|
|
|
/* The following are some testing auxiliary macros. Do not enable them
|
|
|
|
in a production environment. */
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* If this is enabled then lock folds will always be different
|
|
|
|
resulting in equal rows being put in a different cells of the hash
|
|
|
|
table. Checking for duplicates will be flawed because different
|
|
|
|
fold will be calculated when a row is searched in the hash table. */
|
|
|
|
#define TEST_LOCK_FOLD_ALWAYS_DIFFERENT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
2007-09-22 08:36:59 +00:00
|
|
|
/* This effectively kills the search-for-duplicate-before-adding-a-row
|
|
|
|
function, but searching in the hash is still performed. It will always
|
|
|
|
be assumed that lock is not present and insertion will be performed in
|
|
|
|
the hash table. */
|
2007-09-20 11:23:44 +00:00
|
|
|
#define TEST_NO_LOCKS_ROW_IS_EVER_EQUAL_TO_LOCK_T
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* This aggressively repeats adding each row many times. Depending on
|
|
|
|
the above settings this may be noop or may result in lots of rows being
|
|
|
|
added. */
|
|
|
|
#define TEST_ADD_EACH_LOCKS_ROW_MANY_TIMES
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* Very similar to TEST_NO_LOCKS_ROW_IS_EVER_EQUAL_TO_LOCK_T but hash
|
|
|
|
table search is not performed at all. */
|
|
|
|
#define TEST_DO_NOT_CHECK_FOR_DUPLICATE_ROWS
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* Do not insert each row into the hash table, duplicates may appear
|
|
|
|
if this is enabled, also if this is enabled searching into the hash is
|
|
|
|
noop because it will be empty. */
|
|
|
|
#define TEST_DO_NOT_INSERT_INTO_THE_HASH_TABLE
|
|
|
|
#endif
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
#define MAX_ALLOWED_FOR_STORAGE(cache) \
|
|
|
|
(TRX_I_S_MEM_LIMIT \
|
|
|
|
- (cache)->mem_allocd)
|
|
|
|
|
|
|
|
#define MAX_ALLOWED_FOR_ALLOC(cache) \
|
|
|
|
(TRX_I_S_MEM_LIMIT \
|
|
|
|
- (cache)->mem_allocd \
|
|
|
|
- ha_storage_get_size((cache)->storage))
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
/* Memory for each table in the intermediate buffer is allocated in
|
|
|
|
separate chunks. These chunks are considered to be concatenated to
|
|
|
|
represent one flat array of rows. */
|
|
|
|
typedef struct i_s_mem_chunk_struct {
|
|
|
|
ulint offset; /* offset, in number of rows */
|
|
|
|
ulint rows_allocd; /* the size of this chunk, in number
|
|
|
|
of rows */
|
|
|
|
void* base; /* start of the chunk */
|
|
|
|
} i_s_mem_chunk_t;
|
|
|
|
|
|
|
|
/* This represents one table's cache. */
|
|
|
|
typedef struct i_s_table_cache_struct {
|
|
|
|
ullint rows_used; /* number of used rows */
|
|
|
|
ullint rows_allocd; /* number of allocated rows */
|
|
|
|
ulint row_size; /* size of a single row */
|
|
|
|
i_s_mem_chunk_t chunks[MEM_CHUNKS_IN_TABLE_CACHE]; /* array of
|
|
|
|
memory chunks that stores the
|
|
|
|
rows */
|
|
|
|
} i_s_table_cache_t;
|
|
|
|
|
|
|
|
/* This structure describes the intermediate buffer */
|
|
|
|
struct trx_i_s_cache_struct {
|
|
|
|
rw_lock_t rw_lock; /* read-write lock protecting
|
|
|
|
the rest of this structure */
|
|
|
|
ullint last_read; /* last time the cache was read;
|
|
|
|
measured in microseconds since
|
|
|
|
epoch */
|
|
|
|
mutex_t last_read_mutex;/* mutex protecting the
|
|
|
|
last_read member - it is updated
|
|
|
|
inside a shared lock of the
|
|
|
|
rw_lock member */
|
|
|
|
i_s_table_cache_t innodb_trx; /* innodb_trx table */
|
|
|
|
i_s_table_cache_t innodb_locks; /* innodb_locks table */
|
|
|
|
i_s_table_cache_t innodb_lock_waits;/* innodb_lock_waits table */
|
|
|
|
/* the hash table size is LOCKS_HASH_CELLS_NUM * sizeof(void*) bytes */
|
2007-09-24 12:25:06 +00:00
|
|
|
#define LOCKS_HASH_CELLS_NUM 10000
|
2007-09-20 11:23:44 +00:00
|
|
|
hash_table_t* locks_hash; /* hash table used to eliminate
|
|
|
|
duplicate entries in the
|
|
|
|
innodb_locks table */
|
2007-09-24 12:25:06 +00:00
|
|
|
#define CACHE_STORAGE_INITIAL_SIZE 1024
|
|
|
|
#define CACHE_STORAGE_HASH_CELLS 2048
|
|
|
|
ha_storage_t* storage; /* storage for external volatile
|
|
|
|
data that can possibly not be
|
|
|
|
available later, when we release
|
|
|
|
the kernel mutex */
|
2007-11-16 13:12:13 +00:00
|
|
|
ulint mem_allocd; /* the amount of memory
|
|
|
|
allocated with mem_alloc*() */
|
|
|
|
ibool is_truncated; /* this is TRUE if the memory
|
|
|
|
limit was hit and thus the data
|
|
|
|
in the cache is truncated */
|
2007-09-20 11:23:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* This is the intermediate buffer where data needed to fill the
|
|
|
|
INFORMATION SCHEMA tables is fetched and later retrieved by the C++
|
|
|
|
code in handler/i_s.cc. */
|
|
|
|
static trx_i_s_cache_t trx_i_s_cache_static;
|
|
|
|
trx_i_s_cache_t* trx_i_s_cache = &trx_i_s_cache_static;
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
For a record lock that is in waiting state retrieves the only bit that
|
|
|
|
is set, for a table lock returns ULINT_UNDEFINED. */
|
|
|
|
static
|
|
|
|
ulint
|
|
|
|
wait_lock_get_heap_no(
|
|
|
|
/*==================*/
|
|
|
|
/* out: record number within the heap */
|
|
|
|
const lock_t* lock) /* in: lock */
|
|
|
|
{
|
|
|
|
ulint ret;
|
|
|
|
|
|
|
|
switch (lock_get_type(lock)) {
|
|
|
|
case LOCK_REC:
|
|
|
|
ret = lock_rec_find_set_bit(lock);
|
|
|
|
ut_a(ret != ULINT_UNDEFINED);
|
|
|
|
break;
|
|
|
|
case LOCK_TABLE:
|
|
|
|
ret = ULINT_UNDEFINED;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Initializes the members of a table cache. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
table_cache_init(
|
|
|
|
/*=============*/
|
|
|
|
i_s_table_cache_t* table_cache, /* out: table cache */
|
|
|
|
size_t row_size) /* in: the size of a
|
|
|
|
row */
|
|
|
|
{
|
2007-09-22 08:34:26 +00:00
|
|
|
ulint i;
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
table_cache->rows_used = 0;
|
|
|
|
table_cache->rows_allocd = 0;
|
|
|
|
table_cache->row_size = row_size;
|
|
|
|
|
2007-09-22 08:34:26 +00:00
|
|
|
for (i = 0; i < MEM_CHUNKS_IN_TABLE_CACHE; i++) {
|
|
|
|
|
|
|
|
/* the memory is actually allocated in
|
|
|
|
table_cache_create_empty_row() */
|
|
|
|
table_cache->chunks[i].base = NULL;
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Returns an empty row from a table cache. The row is allocated if no more
|
2007-11-16 13:12:13 +00:00
|
|
|
empty rows are available. The number of used rows is incremented.
|
|
|
|
If the memory limit is hit then NULL is returned and nothing is
|
|
|
|
allocated. */
|
2007-09-20 11:23:44 +00:00
|
|
|
static
|
|
|
|
void*
|
|
|
|
table_cache_create_empty_row(
|
|
|
|
/*=========================*/
|
2007-11-16 13:12:13 +00:00
|
|
|
/* out: empty row, or
|
|
|
|
NULL if out of memory */
|
|
|
|
i_s_table_cache_t* table_cache, /* in/out: table cache */
|
|
|
|
trx_i_s_cache_t* cache) /* in/out: cache to record
|
|
|
|
how many bytes are
|
|
|
|
allocated */
|
2007-09-20 11:23:44 +00:00
|
|
|
{
|
|
|
|
ulint i;
|
|
|
|
void* row;
|
|
|
|
|
|
|
|
ut_a(table_cache->rows_used <= table_cache->rows_allocd);
|
|
|
|
|
|
|
|
if (table_cache->rows_used == table_cache->rows_allocd) {
|
|
|
|
|
|
|
|
/* rows_used == rows_allocd means that new chunk needs
|
|
|
|
to be allocated: either no more empty rows in the
|
|
|
|
last allocated chunk or nothing has been allocated yet
|
|
|
|
(rows_num == rows_allocd == 0); */
|
|
|
|
|
2007-09-22 08:34:26 +00:00
|
|
|
i_s_mem_chunk_t* chunk;
|
|
|
|
ulint req_bytes;
|
|
|
|
ulint got_bytes;
|
|
|
|
ulint req_rows;
|
|
|
|
ulint got_rows;
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
/* find the first not allocated chunk */
|
|
|
|
for (i = 0; i < MEM_CHUNKS_IN_TABLE_CACHE; i++) {
|
|
|
|
|
|
|
|
if (table_cache->chunks[i].base == NULL) {
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* i == MEM_CHUNKS_IN_TABLE_CACHE means that all chunks
|
|
|
|
have been allocated :-X */
|
|
|
|
ut_a(i < MEM_CHUNKS_IN_TABLE_CACHE);
|
|
|
|
|
2007-09-22 08:34:26 +00:00
|
|
|
/* allocate the chunk we just found */
|
|
|
|
|
|
|
|
if (i == 0) {
|
|
|
|
|
|
|
|
/* first chunk, nothing is allocated yet */
|
|
|
|
req_rows = TABLE_CACHE_INITIAL_ROWSNUM;
|
|
|
|
} else {
|
|
|
|
|
|
|
|
/* Memory is increased by the formula
|
|
|
|
new = old + old / 2; We are trying not to be
|
|
|
|
aggressive here (= using the common new = old * 2)
|
|
|
|
because the allocated memory will not be freed
|
|
|
|
until InnoDB exit (it is reused). So it is better
|
|
|
|
to once allocate the memory in more steps, but
|
|
|
|
have less unused/wasted memory than to use less
|
|
|
|
steps in allocation (which is done once in a
|
|
|
|
lifetime) but end up with lots of unused/wasted
|
|
|
|
memory. */
|
|
|
|
req_rows = table_cache->rows_allocd / 2;
|
|
|
|
}
|
|
|
|
req_bytes = req_rows * table_cache->row_size;
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
if (req_bytes > MAX_ALLOWED_FOR_ALLOC(cache)) {
|
|
|
|
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
2007-09-22 08:34:26 +00:00
|
|
|
chunk = &table_cache->chunks[i];
|
|
|
|
|
|
|
|
chunk->base = mem_alloc2(req_bytes, &got_bytes);
|
|
|
|
|
|
|
|
got_rows = got_bytes / table_cache->row_size;
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
cache->mem_allocd += got_bytes;
|
|
|
|
|
2007-09-22 08:34:26 +00:00
|
|
|
#if 0
|
|
|
|
printf("allocating chunk %d req bytes=%lu, got bytes=%lu, "
|
|
|
|
"row size=%lu, "
|
|
|
|
"req rows=%lu, got rows=%lu\n",
|
|
|
|
i, req_bytes, got_bytes,
|
|
|
|
table_cache->row_size,
|
|
|
|
req_rows, got_rows);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
chunk->rows_allocd = got_rows;
|
|
|
|
|
|
|
|
table_cache->rows_allocd += got_rows;
|
|
|
|
|
|
|
|
/* adjust the offset of the next chunk */
|
|
|
|
if (i < MEM_CHUNKS_IN_TABLE_CACHE - 1) {
|
|
|
|
|
|
|
|
table_cache->chunks[i + 1].offset
|
|
|
|
= chunk->offset + chunk->rows_allocd;
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
/* return the first empty row in the newly allocated
|
|
|
|
chunk */
|
2007-09-22 08:34:26 +00:00
|
|
|
row = chunk->base;
|
2007-09-20 11:23:44 +00:00
|
|
|
} else {
|
|
|
|
|
|
|
|
char* chunk_start;
|
|
|
|
ulint offset;
|
|
|
|
|
|
|
|
/* there is an empty row, no need to allocate new
|
|
|
|
chunks */
|
|
|
|
|
|
|
|
/* find the first chunk that contains allocated but
|
|
|
|
empty/unused rows */
|
|
|
|
for (i = 0; i < MEM_CHUNKS_IN_TABLE_CACHE; i++) {
|
|
|
|
|
|
|
|
if (table_cache->chunks[i].offset
|
|
|
|
+ table_cache->chunks[i].rows_allocd
|
|
|
|
> table_cache->rows_used) {
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* i == MEM_CHUNKS_IN_TABLE_CACHE means that all chunks
|
|
|
|
are full, but
|
|
|
|
table_cache->rows_used != table_cache->rows_allocd means
|
|
|
|
exactly the opposite - there are allocated but
|
|
|
|
empty/unused rows :-X */
|
|
|
|
ut_a(i < MEM_CHUNKS_IN_TABLE_CACHE);
|
|
|
|
|
|
|
|
chunk_start = (char*) table_cache->chunks[i].base;
|
|
|
|
offset = table_cache->rows_used
|
|
|
|
- table_cache->chunks[i].offset;
|
|
|
|
|
|
|
|
row = chunk_start + offset * table_cache->row_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
table_cache->rows_used++;
|
|
|
|
|
|
|
|
return(row);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
2007-11-16 13:12:13 +00:00
|
|
|
Fills i_s_trx_row_t object.
|
|
|
|
If memory can not be allocated then FALSE is returned. */
|
2007-09-20 11:23:44 +00:00
|
|
|
static
|
2007-11-16 13:12:13 +00:00
|
|
|
ibool
|
2007-09-20 11:23:44 +00:00
|
|
|
fill_trx_row(
|
|
|
|
/*=========*/
|
2007-11-16 13:12:13 +00:00
|
|
|
/* out: FALSE if
|
|
|
|
allocation fails */
|
2007-09-20 11:23:44 +00:00
|
|
|
i_s_trx_row_t* row, /* out: result object
|
|
|
|
that's filled */
|
|
|
|
const trx_t* trx, /* in: transaction to
|
|
|
|
get data from */
|
2007-11-29 13:47:09 +00:00
|
|
|
const i_s_locks_row_t* requested_lock_row,/* in: pointer to the
|
2007-09-20 11:23:44 +00:00
|
|
|
corresponding row in
|
|
|
|
innodb_locks if trx is
|
|
|
|
waiting or NULL if trx
|
|
|
|
is not waiting */
|
2007-11-07 20:38:07 +00:00
|
|
|
trx_i_s_cache_t* cache) /* in/out: cache into
|
|
|
|
which to copy volatile
|
|
|
|
strings */
|
2007-09-20 11:23:44 +00:00
|
|
|
{
|
|
|
|
row->trx_id = trx_get_id(trx);
|
2007-11-28 07:07:23 +00:00
|
|
|
row->trx_weight = (ullint) ut_conv_dulint_to_longlong(TRX_WEIGHT(trx));
|
2007-09-20 11:23:44 +00:00
|
|
|
row->trx_started = (ib_time_t) trx->start_time;
|
|
|
|
row->trx_state = trx_get_que_state_str(trx);
|
|
|
|
|
|
|
|
if (trx->wait_lock != NULL) {
|
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
ut_a(requested_lock_row != NULL);
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
row->requested_lock_row = requested_lock_row;
|
2007-09-20 11:23:44 +00:00
|
|
|
row->trx_wait_started = (ib_time_t) trx->wait_started;
|
|
|
|
} else {
|
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
ut_a(requested_lock_row == NULL);
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
row->requested_lock_row = NULL;
|
2007-09-20 11:23:44 +00:00
|
|
|
row->trx_wait_started = 0;
|
|
|
|
}
|
|
|
|
|
2007-11-07 14:14:33 +00:00
|
|
|
row->trx_mysql_thread_id = ib_thd_get_thread_id(trx->mysql_thd);
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-11-07 20:38:07 +00:00
|
|
|
if (trx->mysql_query_str != NULL && *trx->mysql_query_str != NULL) {
|
|
|
|
|
|
|
|
if (strlen(*trx->mysql_query_str)
|
|
|
|
> TRX_I_S_TRX_QUERY_MAX_LEN) {
|
|
|
|
|
|
|
|
char query[TRX_I_S_TRX_QUERY_MAX_LEN + 1];
|
|
|
|
|
|
|
|
memcpy(query, *trx->mysql_query_str,
|
|
|
|
TRX_I_S_TRX_QUERY_MAX_LEN);
|
|
|
|
query[TRX_I_S_TRX_QUERY_MAX_LEN] = '\0';
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
row->trx_query = ha_storage_put_memlim(
|
2007-11-07 20:38:07 +00:00
|
|
|
cache->storage, query,
|
2007-11-16 13:12:13 +00:00
|
|
|
TRX_I_S_TRX_QUERY_MAX_LEN + 1,
|
|
|
|
MAX_ALLOWED_FOR_STORAGE(cache));
|
2007-11-07 20:38:07 +00:00
|
|
|
} else {
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
row->trx_query = ha_storage_put_str_memlim(
|
|
|
|
cache->storage, *trx->mysql_query_str,
|
|
|
|
MAX_ALLOWED_FOR_STORAGE(cache));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (row->trx_query == NULL) {
|
|
|
|
|
|
|
|
return(FALSE);
|
2007-11-07 20:38:07 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
|
|
|
|
row->trx_query = NULL;
|
|
|
|
}
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
return(TRUE);
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
2007-10-29 13:03:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
Format the nth field of "rec" and put it in "buf". The result is always
|
|
|
|
'\0'-terminated. Returns the number of bytes that were written to "buf"
|
|
|
|
(including the terminating '\0'). */
|
|
|
|
static
|
|
|
|
ulint
|
|
|
|
put_nth_field(
|
|
|
|
/*==========*/
|
|
|
|
/* out: end of the result */
|
|
|
|
char* buf, /* out: buffer */
|
|
|
|
ulint buf_size,/* in: buffer size in bytes */
|
|
|
|
ulint n, /* in: number of field */
|
|
|
|
const dict_index_t* index, /* in: index */
|
|
|
|
const rec_t* rec, /* in: record */
|
|
|
|
const ulint* offsets)/* in: record offsets, returned
|
|
|
|
by rec_get_offsets() */
|
|
|
|
{
|
|
|
|
const byte* data;
|
|
|
|
ulint data_len;
|
|
|
|
dict_field_t* dict_field;
|
|
|
|
ulint ret;
|
|
|
|
|
|
|
|
ut_ad(rec_offs_validate(rec, NULL, offsets));
|
|
|
|
|
|
|
|
if (buf_size == 0) {
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
if (n > 0) {
|
|
|
|
/* we must append ", " before the actual data */
|
|
|
|
|
|
|
|
if (buf_size < 3) {
|
|
|
|
|
|
|
|
buf[0] = '\0';
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(buf, ", ", 3);
|
|
|
|
|
|
|
|
buf += 2;
|
|
|
|
buf_size -= 2;
|
|
|
|
ret += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* now buf_size >= 1 */
|
|
|
|
|
|
|
|
data = rec_get_nth_field(rec, offsets, n, &data_len);
|
|
|
|
|
|
|
|
dict_field = dict_index_get_nth_field(index, n);
|
|
|
|
|
|
|
|
ret += row_raw_format((const char*) data, data_len,
|
|
|
|
dict_field, buf, buf_size);
|
|
|
|
|
|
|
|
return(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
2007-11-16 13:12:13 +00:00
|
|
|
Fills the "lock_data" member of i_s_locks_row_t object.
|
|
|
|
If memory can not be allocated then FALSE is returned. */
|
2007-10-29 13:03:53 +00:00
|
|
|
static
|
2007-11-16 13:12:13 +00:00
|
|
|
ibool
|
2007-10-29 13:03:53 +00:00
|
|
|
fill_lock_data(
|
|
|
|
/*===========*/
|
2007-11-16 13:12:13 +00:00
|
|
|
/* out: FALSE if allocation fails */
|
2007-10-29 13:03:53 +00:00
|
|
|
const char** lock_data,/* out: "lock_data" to fill */
|
|
|
|
const lock_t* lock, /* in: lock used to find the data */
|
|
|
|
ulint heap_no,/* in: rec num used to find the data */
|
|
|
|
trx_i_s_cache_t* cache) /* in/out: cache where to store
|
|
|
|
volatile data */
|
|
|
|
{
|
|
|
|
mtr_t mtr;
|
|
|
|
|
|
|
|
const buf_block_t* block;
|
|
|
|
const page_t* page;
|
|
|
|
const rec_t* rec;
|
|
|
|
|
|
|
|
ut_a(lock_get_type(lock) == LOCK_REC);
|
|
|
|
|
|
|
|
mtr_start(&mtr);
|
|
|
|
|
|
|
|
block = buf_page_try_get(lock_rec_get_space_id(lock),
|
|
|
|
lock_rec_get_page_no(lock),
|
|
|
|
&mtr);
|
|
|
|
|
|
|
|
if (block == NULL) {
|
|
|
|
|
|
|
|
*lock_data = NULL;
|
|
|
|
|
|
|
|
mtr_commit(&mtr);
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
return(TRUE);
|
2007-10-29 13:03:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
page = (const page_t*) buf_block_get_frame(block);
|
|
|
|
|
|
|
|
rec = page_find_rec_with_heap_no(page, heap_no);
|
|
|
|
|
|
|
|
if (page_rec_is_infimum(rec)) {
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
*lock_data = ha_storage_put_str_memlim(
|
|
|
|
cache->storage, "infimum pseudo-record",
|
|
|
|
MAX_ALLOWED_FOR_STORAGE(cache));
|
2007-10-29 13:03:53 +00:00
|
|
|
} else if (page_rec_is_supremum(rec)) {
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
*lock_data = ha_storage_put_str_memlim(
|
|
|
|
cache->storage, "supremum pseudo-record",
|
|
|
|
MAX_ALLOWED_FOR_STORAGE(cache));
|
2007-10-29 13:03:53 +00:00
|
|
|
} else {
|
|
|
|
|
|
|
|
const dict_index_t* index;
|
|
|
|
ulint n_fields;
|
|
|
|
mem_heap_t* heap;
|
|
|
|
ulint offsets_onstack[REC_OFFS_NORMAL_SIZE];
|
|
|
|
ulint* offsets;
|
|
|
|
char buf[TRX_I_S_LOCK_DATA_MAX_LEN];
|
|
|
|
ulint buf_used;
|
|
|
|
ulint i;
|
|
|
|
|
|
|
|
rec_offs_init(offsets_onstack);
|
|
|
|
offsets = offsets_onstack;
|
|
|
|
|
|
|
|
index = lock_rec_get_index(lock);
|
|
|
|
|
|
|
|
n_fields = dict_index_get_n_unique(index);
|
|
|
|
|
|
|
|
ut_a(n_fields > 0);
|
|
|
|
|
|
|
|
heap = NULL;
|
|
|
|
offsets = rec_get_offsets(rec, index, offsets, n_fields,
|
|
|
|
&heap);
|
|
|
|
|
|
|
|
/* format and store the data */
|
|
|
|
|
|
|
|
buf_used = 0;
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
|
|
|
|
|
|
buf_used += put_nth_field(
|
|
|
|
buf + buf_used, sizeof(buf) - buf_used,
|
|
|
|
i, index, rec, offsets) - 1;
|
|
|
|
}
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
*lock_data = (const char*) ha_storage_put_memlim(
|
|
|
|
cache->storage, buf, buf_used + 1,
|
|
|
|
MAX_ALLOWED_FOR_STORAGE(cache));
|
2007-10-29 13:03:53 +00:00
|
|
|
|
|
|
|
if (UNIV_UNLIKELY(heap != NULL)) {
|
|
|
|
|
|
|
|
/* this means that rec_get_offsets() has created a new
|
|
|
|
heap and has stored offsets in it; check that this is
|
|
|
|
really the case and free the heap */
|
|
|
|
ut_a(offsets != offsets_onstack);
|
|
|
|
mem_heap_free(heap);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mtr_commit(&mtr);
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
if (*lock_data == NULL) {
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return(TRUE);
|
2007-10-29 13:03:53 +00:00
|
|
|
}
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
/***********************************************************************
|
2007-11-16 13:12:13 +00:00
|
|
|
Fills i_s_locks_row_t object. Returns its first argument.
|
|
|
|
If memory can not be allocated then FALSE is returned. */
|
2007-09-20 11:23:44 +00:00
|
|
|
static
|
2007-11-16 13:12:13 +00:00
|
|
|
ibool
|
2007-09-20 11:23:44 +00:00
|
|
|
fill_locks_row(
|
|
|
|
/*===========*/
|
2007-11-16 13:12:13 +00:00
|
|
|
/* out: FALSE if allocation fails */
|
2007-09-20 11:23:44 +00:00
|
|
|
i_s_locks_row_t* row, /* out: result object that's filled */
|
|
|
|
const lock_t* lock, /* in: lock to get data from */
|
2007-09-24 12:25:06 +00:00
|
|
|
ulint heap_no,/* in: lock's record number
|
2007-09-20 11:23:44 +00:00
|
|
|
or ULINT_UNDEFINED if the lock
|
|
|
|
is a table lock */
|
2007-09-24 12:25:06 +00:00
|
|
|
trx_i_s_cache_t* cache) /* in/out: cache into which to copy
|
|
|
|
volatile strings */
|
2007-09-20 11:23:44 +00:00
|
|
|
{
|
|
|
|
row->lock_trx_id = lock_get_trx_id(lock);
|
|
|
|
row->lock_mode = lock_get_mode_str(lock);
|
|
|
|
row->lock_type = lock_get_type_str(lock);
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
row->lock_table = ha_storage_put_str_memlim(
|
|
|
|
cache->storage, lock_get_table_name(lock),
|
|
|
|
MAX_ALLOWED_FOR_STORAGE(cache));
|
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
if (row->lock_table == NULL) {
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
switch (lock_get_type(lock)) {
|
|
|
|
case LOCK_REC:
|
2007-11-16 13:12:13 +00:00
|
|
|
row->lock_index = ha_storage_put_str_memlim(
|
|
|
|
cache->storage, lock_rec_get_index_name(lock),
|
|
|
|
MAX_ALLOWED_FOR_STORAGE(cache));
|
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
if (row->lock_index == NULL) {
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
row->lock_space = lock_rec_get_space_id(lock);
|
|
|
|
row->lock_page = lock_rec_get_page_no(lock);
|
|
|
|
row->lock_rec = heap_no;
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
if (!fill_lock_data(&row->lock_data, lock, heap_no, cache)) {
|
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2007-10-29 13:03:53 +00:00
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
break;
|
|
|
|
case LOCK_TABLE:
|
|
|
|
row->lock_index = NULL;
|
|
|
|
|
|
|
|
row->lock_space = ULINT_UNDEFINED;
|
|
|
|
row->lock_page = ULINT_UNDEFINED;
|
|
|
|
row->lock_rec = ULINT_UNDEFINED;
|
|
|
|
|
2007-10-29 13:03:53 +00:00
|
|
|
row->lock_data = NULL;
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
row->lock_table_id = lock_get_table_id(lock);
|
|
|
|
|
|
|
|
row->hash_chain.value = row;
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
return(TRUE);
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Fills i_s_lock_waits_row_t object. Returns its first argument. */
|
|
|
|
static
|
|
|
|
i_s_lock_waits_row_t*
|
|
|
|
fill_lock_waits_row(
|
|
|
|
/*================*/
|
|
|
|
/* out: result object
|
|
|
|
that's filled */
|
|
|
|
i_s_lock_waits_row_t* row, /* out: result object
|
|
|
|
that's filled */
|
2007-11-29 13:47:09 +00:00
|
|
|
const i_s_locks_row_t* requested_lock_row,/* in: pointer to the
|
|
|
|
relevant requested lock
|
2007-09-20 11:23:44 +00:00
|
|
|
row in innodb_locks */
|
2007-11-29 13:47:09 +00:00
|
|
|
const i_s_locks_row_t* blocking_lock_row)/* in: pointer to the
|
|
|
|
relevant blocking lock
|
2007-09-20 11:23:44 +00:00
|
|
|
row in innodb_locks */
|
|
|
|
{
|
2007-11-29 13:47:09 +00:00
|
|
|
row->requested_lock_row = requested_lock_row;
|
|
|
|
row->blocking_lock_row = blocking_lock_row;
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
return(row);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Calculates a hash fold for a lock. For a record lock the fold is
|
|
|
|
calculated from 4 elements, which uniquely identify a lock at a given
|
|
|
|
point in time: transaction id, space id, page number, record number.
|
|
|
|
For a table lock the fold is table's id. */
|
|
|
|
static
|
|
|
|
ulint
|
|
|
|
fold_lock(
|
|
|
|
/*======*/
|
|
|
|
/* out: fold */
|
|
|
|
const lock_t* lock, /* in: lock object to fold */
|
|
|
|
ulint heap_no)/* in: lock's record number
|
|
|
|
or ULINT_UNDEFINED if the lock
|
|
|
|
is a table lock */
|
|
|
|
{
|
|
|
|
#ifdef TEST_LOCK_FOLD_ALWAYS_DIFFERENT
|
|
|
|
static ulint fold = 0;
|
|
|
|
|
|
|
|
return(fold++);
|
|
|
|
#else
|
|
|
|
ulint ret;
|
|
|
|
|
|
|
|
switch (lock_get_type(lock)) {
|
|
|
|
case LOCK_REC:
|
|
|
|
ut_a(heap_no != ULINT_UNDEFINED);
|
|
|
|
|
|
|
|
ret = ut_fold_ulint_pair((ulint) lock_get_trx_id(lock),
|
|
|
|
lock_rec_get_space_id(lock));
|
|
|
|
|
|
|
|
ret = ut_fold_ulint_pair(ret,
|
|
|
|
lock_rec_get_page_no(lock));
|
|
|
|
|
|
|
|
ret = ut_fold_ulint_pair(ret, heap_no);
|
|
|
|
|
|
|
|
break;
|
|
|
|
case LOCK_TABLE:
|
|
|
|
/* this check is actually not necessary for continuing
|
|
|
|
correct operation, but something must have gone wrong if
|
|
|
|
it fails. */
|
|
|
|
ut_a(heap_no == ULINT_UNDEFINED);
|
|
|
|
|
|
|
|
ret = (ulint) lock_get_table_id(lock);
|
|
|
|
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return(ret);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Checks whether i_s_locks_row_t object represents a lock_t object. */
|
|
|
|
static
|
|
|
|
ibool
|
|
|
|
locks_row_eq_lock(
|
|
|
|
/*==============*/
|
|
|
|
/* out: TRUE if they match */
|
|
|
|
const i_s_locks_row_t* row, /* in: innodb_locks row */
|
|
|
|
const lock_t* lock, /* in: lock object */
|
|
|
|
ulint heap_no)/* in: lock's record number
|
|
|
|
or ULINT_UNDEFINED if the lock
|
|
|
|
is a table lock */
|
|
|
|
{
|
|
|
|
#ifdef TEST_NO_LOCKS_ROW_IS_EVER_EQUAL_TO_LOCK_T
|
|
|
|
return(0);
|
|
|
|
#else
|
|
|
|
switch (lock_get_type(lock)) {
|
|
|
|
case LOCK_REC:
|
|
|
|
ut_a(heap_no != ULINT_UNDEFINED);
|
|
|
|
|
|
|
|
return(row->lock_trx_id == lock_get_trx_id(lock)
|
|
|
|
&& row->lock_space == lock_rec_get_space_id(lock)
|
|
|
|
&& row->lock_page == lock_rec_get_page_no(lock)
|
|
|
|
&& row->lock_rec == heap_no);
|
|
|
|
|
|
|
|
case LOCK_TABLE:
|
|
|
|
/* this check is actually not necessary for continuing
|
|
|
|
correct operation, but something must have gone wrong if
|
|
|
|
it fails. */
|
|
|
|
ut_a(heap_no == ULINT_UNDEFINED);
|
|
|
|
|
|
|
|
return(row->lock_trx_id == lock_get_trx_id(lock)
|
|
|
|
&& row->lock_table_id == lock_get_table_id(lock));
|
|
|
|
|
|
|
|
default:
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Searches for a row in the innodb_locks cache that has a specified id.
|
|
|
|
This happens in O(1) time since a hash table is used. Returns pointer to
|
|
|
|
the row or NULL if none is found. */
|
|
|
|
static
|
|
|
|
i_s_locks_row_t*
|
|
|
|
search_innodb_locks(
|
|
|
|
/*================*/
|
|
|
|
/* out: row or NULL */
|
|
|
|
trx_i_s_cache_t* cache, /* in: cache */
|
|
|
|
const lock_t* lock, /* in: lock to search for */
|
|
|
|
ulint heap_no)/* in: lock's record number
|
|
|
|
or ULINT_UNDEFINED if the lock
|
|
|
|
is a table lock */
|
|
|
|
{
|
|
|
|
i_s_hash_chain_t* hash_chain;
|
|
|
|
|
|
|
|
HASH_SEARCH(
|
|
|
|
/* hash_chain->"next" */
|
|
|
|
next,
|
|
|
|
/* the hash table */
|
|
|
|
cache->locks_hash,
|
|
|
|
/* fold */
|
|
|
|
fold_lock(lock, heap_no),
|
|
|
|
/* the type of the next variable */
|
|
|
|
i_s_hash_chain_t*,
|
|
|
|
/* auxiliary variable */
|
|
|
|
hash_chain,
|
|
|
|
/* this determines if we have found the lock */
|
|
|
|
locks_row_eq_lock(hash_chain->value, lock, heap_no));
|
|
|
|
|
|
|
|
if (hash_chain == NULL) {
|
|
|
|
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
/* else */
|
|
|
|
|
|
|
|
return(hash_chain->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Adds new element to the locks cache, enlarging it if necessary.
|
|
|
|
Returns a pointer to the added row. If the row is already present then
|
2007-11-16 13:12:13 +00:00
|
|
|
no row is added and a pointer to the existing row is returned.
|
|
|
|
If row can not be allocated then NULL is returned. */
|
2007-09-20 11:23:44 +00:00
|
|
|
static
|
|
|
|
i_s_locks_row_t*
|
|
|
|
add_lock_to_cache(
|
|
|
|
/*==============*/
|
|
|
|
/* out: row */
|
|
|
|
trx_i_s_cache_t* cache, /* in/out: cache */
|
|
|
|
const lock_t* lock, /* in: the element to add */
|
|
|
|
ulint heap_no)/* in: lock's record number
|
|
|
|
or ULINT_UNDEFINED if the lock
|
|
|
|
is a table lock */
|
|
|
|
{
|
|
|
|
i_s_locks_row_t* dst_row;
|
|
|
|
|
|
|
|
#ifdef TEST_ADD_EACH_LOCKS_ROW_MANY_TIMES
|
|
|
|
ulint i;
|
|
|
|
for (i = 0; i < 10000; i++) {
|
|
|
|
#endif
|
|
|
|
#ifndef TEST_DO_NOT_CHECK_FOR_DUPLICATE_ROWS
|
|
|
|
/* quit if this lock is already present */
|
|
|
|
dst_row = search_innodb_locks(cache, lock, heap_no);
|
|
|
|
if (dst_row != NULL) {
|
|
|
|
|
|
|
|
return(dst_row);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
dst_row = (i_s_locks_row_t*)
|
2007-11-16 13:12:13 +00:00
|
|
|
table_cache_create_empty_row(&cache->innodb_locks, cache);
|
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
if (dst_row == NULL) {
|
|
|
|
|
|
|
|
return(NULL);
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
if (!fill_locks_row(dst_row, lock, heap_no, cache)) {
|
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
cache->innodb_locks.rows_used--;
|
|
|
|
return(NULL);
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
#ifndef TEST_DO_NOT_INSERT_INTO_THE_HASH_TABLE
|
|
|
|
HASH_INSERT(
|
|
|
|
/* the type used in the hash chain */
|
|
|
|
i_s_hash_chain_t,
|
|
|
|
/* hash_chain->"next" */
|
|
|
|
next,
|
|
|
|
/* the hash table */
|
|
|
|
cache->locks_hash,
|
|
|
|
/* fold */
|
|
|
|
fold_lock(lock, heap_no),
|
|
|
|
/* add this data to the hash */
|
|
|
|
&dst_row->hash_chain);
|
|
|
|
#endif
|
|
|
|
#ifdef TEST_ADD_EACH_LOCKS_ROW_MANY_TIMES
|
|
|
|
} /* for()-loop */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return(dst_row);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
2007-11-16 13:12:13 +00:00
|
|
|
Adds new pair of locks to the lock waits cache.
|
|
|
|
If memory can not be allocated then FALSE is returned. */
|
2007-09-20 11:23:44 +00:00
|
|
|
static
|
2007-11-16 13:12:13 +00:00
|
|
|
ibool
|
2007-09-20 11:23:44 +00:00
|
|
|
add_lock_wait_to_cache(
|
|
|
|
/*===================*/
|
2007-11-16 13:12:13 +00:00
|
|
|
/* out: FALSE if
|
|
|
|
allocation fails */
|
2007-09-20 11:23:44 +00:00
|
|
|
trx_i_s_cache_t* cache, /* in/out: cache */
|
2007-11-29 13:47:09 +00:00
|
|
|
const i_s_locks_row_t* requested_lock_row,/* in: pointer to the
|
|
|
|
relevant requested lock
|
2007-09-20 11:23:44 +00:00
|
|
|
row in innodb_locks */
|
2007-11-29 13:47:09 +00:00
|
|
|
const i_s_locks_row_t* blocking_lock_row)/* in: pointer to the
|
|
|
|
relevant blocking lock
|
2007-09-20 11:23:44 +00:00
|
|
|
row in innodb_locks */
|
|
|
|
{
|
|
|
|
i_s_lock_waits_row_t* dst_row;
|
|
|
|
|
|
|
|
dst_row = (i_s_lock_waits_row_t*)
|
2007-11-16 13:12:13 +00:00
|
|
|
table_cache_create_empty_row(&cache->innodb_lock_waits,
|
|
|
|
cache);
|
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
if (dst_row == NULL) {
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
fill_lock_waits_row(dst_row, requested_lock_row, blocking_lock_row);
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
return(TRUE);
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Adds transaction's relevant (important) locks to cache.
|
|
|
|
If the transaction is waiting, then the wait lock is added to
|
|
|
|
innodb_locks and a pointer to the added row is returned in
|
2007-11-29 13:47:09 +00:00
|
|
|
requested_lock_row, otherwise requested_lock_row is set to NULL.
|
2007-11-16 13:12:13 +00:00
|
|
|
If rows can not be allocated then FALSE is returned and the value of
|
2007-11-29 13:47:09 +00:00
|
|
|
requested_lock_row is undefined. */
|
2007-09-20 11:23:44 +00:00
|
|
|
static
|
2007-11-16 13:12:13 +00:00
|
|
|
ibool
|
2007-09-20 11:23:44 +00:00
|
|
|
add_trx_relevant_locks_to_cache(
|
|
|
|
/*============================*/
|
2007-11-16 13:12:13 +00:00
|
|
|
/* out: FALSE if allocation fails */
|
2007-09-20 11:23:44 +00:00
|
|
|
trx_i_s_cache_t* cache, /* in/out: cache */
|
|
|
|
const trx_t* trx, /* in: transaction */
|
2007-11-29 13:47:09 +00:00
|
|
|
i_s_locks_row_t** requested_lock_row)/* out: pointer to the
|
|
|
|
requested lock row, or NULL or
|
|
|
|
undefined */
|
2007-09-20 11:23:44 +00:00
|
|
|
{
|
|
|
|
/* If transaction is waiting we add the wait lock and all locks
|
|
|
|
from another transactions that are blocking the wait lock. */
|
|
|
|
if (trx->que_state == TRX_QUE_LOCK_WAIT) {
|
|
|
|
|
|
|
|
const lock_t* curr_lock;
|
|
|
|
ulint wait_lock_heap_no;
|
2007-11-29 13:47:09 +00:00
|
|
|
i_s_locks_row_t* blocking_lock_row;
|
2007-09-20 11:23:44 +00:00
|
|
|
lock_queue_iterator_t iter;
|
|
|
|
|
|
|
|
ut_a(trx->wait_lock != NULL);
|
|
|
|
|
|
|
|
wait_lock_heap_no
|
|
|
|
= wait_lock_get_heap_no(trx->wait_lock);
|
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
/* add the requested lock */
|
|
|
|
*requested_lock_row
|
2007-09-20 11:23:44 +00:00
|
|
|
= add_lock_to_cache(cache, trx->wait_lock,
|
|
|
|
wait_lock_heap_no);
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
/* memory could not be allocated */
|
2007-11-29 13:47:09 +00:00
|
|
|
if (*requested_lock_row == NULL) {
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
/* then iterate over the locks before the wait lock and
|
|
|
|
add the ones that are blocking it */
|
|
|
|
|
|
|
|
lock_queue_iterator_reset(&iter, trx->wait_lock,
|
|
|
|
ULINT_UNDEFINED);
|
|
|
|
|
|
|
|
curr_lock = lock_queue_iterator_get_prev(&iter);
|
|
|
|
while (curr_lock != NULL) {
|
|
|
|
|
|
|
|
if (lock_has_to_wait(trx->wait_lock,
|
|
|
|
curr_lock)) {
|
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
/* add the lock that is
|
|
|
|
blocking trx->wait_lock */
|
|
|
|
blocking_lock_row
|
2007-09-20 11:23:44 +00:00
|
|
|
= add_lock_to_cache(
|
|
|
|
cache, curr_lock,
|
|
|
|
/* heap_no is the same
|
|
|
|
for the wait and waited
|
|
|
|
locks */
|
|
|
|
wait_lock_heap_no);
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
/* memory could not be allocated */
|
2007-11-29 13:47:09 +00:00
|
|
|
if (blocking_lock_row == NULL) {
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
/* add the relation between both locks
|
|
|
|
to innodb_lock_waits */
|
2007-11-16 13:12:13 +00:00
|
|
|
if (!add_lock_wait_to_cache(
|
2007-11-29 13:47:09 +00:00
|
|
|
cache, *requested_lock_row,
|
|
|
|
blocking_lock_row)) {
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
curr_lock = lock_queue_iterator_get_prev(&iter);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
*requested_lock_row = NULL;
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
return(TRUE);
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Checks if the cache can safely be updated. */
|
|
|
|
static
|
|
|
|
ibool
|
|
|
|
can_cache_be_updated(
|
|
|
|
/*=================*/
|
2007-09-28 04:58:16 +00:00
|
|
|
trx_i_s_cache_t* cache) /* in: cache */
|
2007-09-20 11:23:44 +00:00
|
|
|
{
|
|
|
|
ullint now;
|
|
|
|
|
|
|
|
/* The minimum time that a cache must not be updated after it has been
|
|
|
|
read for the last time; measured in microseconds. We use this technique
|
|
|
|
to ensure that SELECTs which join several INFORMATION SCHEMA tables read
|
|
|
|
the same version of the cache. */
|
|
|
|
#define CACHE_MIN_IDLE_TIME_US 100000 /* 0.1 sec */
|
|
|
|
|
|
|
|
/* Here we read cache->last_read without acquiring its mutex
|
|
|
|
because last_read is only updated when a shared rw lock on the
|
|
|
|
whole cache is being held (see trx_i_s_cache_end_read()) and
|
|
|
|
we are currently holding an exclusive rw lock on the cache.
|
|
|
|
So it is not possible for last_read to be updated while we are
|
|
|
|
reading it. */
|
|
|
|
|
|
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
|
|
ut_a(rw_lock_own(&cache->rw_lock, RW_LOCK_EX));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
now = ut_time_us(NULL);
|
|
|
|
if (now - cache->last_read > CACHE_MIN_IDLE_TIME_US) {
|
|
|
|
|
|
|
|
return(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
2007-09-22 09:18:59 +00:00
|
|
|
/***********************************************************************
|
|
|
|
Declare a cache empty, preparing it to be filled up. Not all resources
|
|
|
|
are freed because they can be reused. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
trx_i_s_cache_clear(
|
|
|
|
/*================*/
|
|
|
|
trx_i_s_cache_t* cache) /* out: cache to clear */
|
|
|
|
{
|
|
|
|
cache->innodb_trx.rows_used = 0;
|
|
|
|
cache->innodb_locks.rows_used = 0;
|
|
|
|
cache->innodb_lock_waits.rows_used = 0;
|
2007-09-24 12:25:06 +00:00
|
|
|
|
2007-09-22 09:18:59 +00:00
|
|
|
hash_table_clear(cache->locks_hash);
|
2007-09-24 12:25:06 +00:00
|
|
|
|
|
|
|
ha_storage_empty(&cache->storage);
|
2007-09-22 09:18:59 +00:00
|
|
|
}
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
/***********************************************************************
|
|
|
|
Fetches the data needed to fill the 3 INFORMATION SCHEMA tables into the
|
|
|
|
table cache buffer. Cache must be locked for write. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
fetch_data_into_cache(
|
|
|
|
/*==================*/
|
|
|
|
trx_i_s_cache_t* cache) /* in/out: cache */
|
|
|
|
{
|
|
|
|
trx_t* trx;
|
|
|
|
i_s_trx_row_t* trx_row;
|
2007-11-29 13:47:09 +00:00
|
|
|
i_s_locks_row_t* requested_lock_row;
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-09-22 09:18:59 +00:00
|
|
|
trx_i_s_cache_clear(cache);
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
/* We iterate over the list of all transactions and add each one
|
|
|
|
to innodb_trx's cache. We also add all locks that are relevant
|
|
|
|
to each transaction into innodb_locks' and innodb_lock_waits'
|
|
|
|
caches. */
|
|
|
|
|
|
|
|
for (trx = UT_LIST_GET_FIRST(trx_sys->trx_list);
|
|
|
|
trx != NULL;
|
|
|
|
trx = UT_LIST_GET_NEXT(trx_list, trx)) {
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
if (!add_trx_relevant_locks_to_cache(cache, trx,
|
2007-11-29 13:47:09 +00:00
|
|
|
&requested_lock_row)) {
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
cache->is_truncated = TRUE;
|
|
|
|
return;
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
|
|
|
trx_row = (i_s_trx_row_t*)
|
2007-11-16 13:12:13 +00:00
|
|
|
table_cache_create_empty_row(&cache->innodb_trx,
|
|
|
|
cache);
|
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
if (trx_row == NULL) {
|
|
|
|
|
|
|
|
cache->is_truncated = TRUE;
|
|
|
|
return;
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
|
2007-11-29 13:47:09 +00:00
|
|
|
if (!fill_trx_row(trx_row, trx, requested_lock_row, cache)) {
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
/* memory could not be allocated */
|
|
|
|
cache->innodb_trx.rows_used--;
|
|
|
|
cache->is_truncated = TRUE;
|
|
|
|
return;
|
|
|
|
}
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
cache->is_truncated = FALSE;
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Update the transactions cache if it has not been read for some time.
|
|
|
|
Called from handler/i_s.cc. */
|
|
|
|
|
|
|
|
int
|
|
|
|
trx_i_s_possibly_fetch_data_into_cache(
|
|
|
|
/*===================================*/
|
|
|
|
/* out: 0 - fetched, 1 - not */
|
|
|
|
trx_i_s_cache_t* cache) /* in/out: cache */
|
|
|
|
{
|
|
|
|
if (!can_cache_be_updated(cache)) {
|
|
|
|
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We need to read trx_sys and record/table lock queues */
|
|
|
|
mutex_enter(&kernel_mutex);
|
|
|
|
|
|
|
|
fetch_data_into_cache(cache);
|
|
|
|
|
|
|
|
mutex_exit(&kernel_mutex);
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
2007-11-16 13:12:13 +00:00
|
|
|
/***********************************************************************
|
|
|
|
Returns TRUE if the data in the cache is truncated due to the memory
|
|
|
|
limit posed by TRX_I_S_MEM_LIMIT. */
|
|
|
|
|
|
|
|
ibool
|
|
|
|
trx_i_s_cache_is_truncated(
|
|
|
|
/*=======================*/
|
|
|
|
/* out: TRUE if truncated */
|
|
|
|
trx_i_s_cache_t* cache) /* in: cache */
|
|
|
|
{
|
|
|
|
return(cache->is_truncated);
|
|
|
|
}
|
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
/***********************************************************************
|
|
|
|
Initialize INFORMATION SCHEMA trx related cache. */
|
|
|
|
|
|
|
|
void
|
|
|
|
trx_i_s_cache_init(
|
|
|
|
/*===============*/
|
|
|
|
trx_i_s_cache_t* cache) /* out: cache to init */
|
|
|
|
{
|
2007-10-12 11:20:13 +00:00
|
|
|
/* The latching is done in the following order:
|
|
|
|
acquire trx_i_s_cache_t::rw_lock, X
|
|
|
|
acquire kernel_mutex
|
|
|
|
release kernel_mutex
|
|
|
|
release trx_i_s_cache_t::rw_lock
|
|
|
|
acquire trx_i_s_cache_t::rw_lock, S
|
|
|
|
acquire trx_i_s_cache_t::last_read_mutex
|
|
|
|
release trx_i_s_cache_t::last_read_mutex
|
|
|
|
release trx_i_s_cache_t::rw_lock */
|
|
|
|
|
|
|
|
rw_lock_create(&cache->rw_lock, SYNC_TRX_I_S_RWLOCK);
|
2007-09-22 09:03:29 +00:00
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
cache->last_read = 0;
|
|
|
|
|
2007-10-12 11:20:13 +00:00
|
|
|
mutex_create(&cache->last_read_mutex, SYNC_TRX_I_S_LAST_READ);
|
2007-09-22 09:03:29 +00:00
|
|
|
|
2007-09-20 11:23:44 +00:00
|
|
|
table_cache_init(&cache->innodb_trx, sizeof(i_s_trx_row_t));
|
|
|
|
table_cache_init(&cache->innodb_locks, sizeof(i_s_locks_row_t));
|
|
|
|
table_cache_init(&cache->innodb_lock_waits,
|
|
|
|
sizeof(i_s_lock_waits_row_t));
|
|
|
|
|
|
|
|
cache->locks_hash = hash_create(LOCKS_HASH_CELLS_NUM);
|
2007-09-24 12:25:06 +00:00
|
|
|
|
|
|
|
cache->storage = ha_storage_create(CACHE_STORAGE_INITIAL_SIZE,
|
|
|
|
CACHE_STORAGE_HASH_CELLS);
|
2007-11-16 13:12:13 +00:00
|
|
|
|
|
|
|
cache->mem_allocd = 0;
|
|
|
|
|
|
|
|
cache->is_truncated = FALSE;
|
2007-09-20 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Issue a shared/read lock on the tables cache. */
|
|
|
|
|
|
|
|
void
|
|
|
|
trx_i_s_cache_start_read(
|
|
|
|
/*=====================*/
|
|
|
|
trx_i_s_cache_t* cache) /* in: cache */
|
|
|
|
{
|
|
|
|
rw_lock_s_lock(&cache->rw_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Release a shared/read lock on the tables cache. */
|
|
|
|
|
|
|
|
void
|
|
|
|
trx_i_s_cache_end_read(
|
|
|
|
/*===================*/
|
|
|
|
trx_i_s_cache_t* cache) /* in: cache */
|
|
|
|
{
|
|
|
|
ullint now;
|
|
|
|
|
|
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
|
|
ut_a(rw_lock_own(&cache->rw_lock, RW_LOCK_SHARED));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* update cache last read time */
|
|
|
|
now = ut_time_us(NULL);
|
|
|
|
mutex_enter(&cache->last_read_mutex);
|
|
|
|
cache->last_read = now;
|
|
|
|
mutex_exit(&cache->last_read_mutex);
|
|
|
|
|
|
|
|
rw_lock_s_unlock(&cache->rw_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Issue an exclusive/write lock on the tables cache. */
|
|
|
|
|
|
|
|
void
|
|
|
|
trx_i_s_cache_start_write(
|
|
|
|
/*======================*/
|
|
|
|
trx_i_s_cache_t* cache) /* in: cache */
|
|
|
|
{
|
|
|
|
rw_lock_x_lock(&cache->rw_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Release an exclusive/write lock on the tables cache. */
|
|
|
|
|
|
|
|
void
|
|
|
|
trx_i_s_cache_end_write(
|
|
|
|
/*====================*/
|
|
|
|
trx_i_s_cache_t* cache) /* in: cache */
|
|
|
|
{
|
|
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
|
|
ut_a(rw_lock_own(&cache->rw_lock, RW_LOCK_EX));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
rw_lock_x_unlock(&cache->rw_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Selects a INFORMATION SCHEMA table cache from the whole cache. */
|
|
|
|
static
|
|
|
|
i_s_table_cache_t*
|
|
|
|
cache_select_table(
|
|
|
|
/*===============*/
|
|
|
|
/* out: table cache */
|
|
|
|
trx_i_s_cache_t* cache, /* in: whole cache */
|
|
|
|
enum i_s_table table) /* in: which table */
|
|
|
|
{
|
|
|
|
i_s_table_cache_t* table_cache;
|
|
|
|
|
|
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
|
|
ut_a(rw_lock_own(&cache->rw_lock, RW_LOCK_SHARED)
|
|
|
|
|| rw_lock_own(&cache->rw_lock, RW_LOCK_EX));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
switch (table) {
|
|
|
|
case I_S_INNODB_TRX:
|
|
|
|
table_cache = &cache->innodb_trx;
|
|
|
|
break;
|
|
|
|
case I_S_INNODB_LOCKS:
|
|
|
|
table_cache = &cache->innodb_locks;
|
|
|
|
break;
|
|
|
|
case I_S_INNODB_LOCK_WAITS:
|
|
|
|
table_cache = &cache->innodb_lock_waits;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return(table_cache);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Retrieves the number of used rows in the cache for a given
|
|
|
|
INFORMATION SCHEMA table. */
|
|
|
|
|
|
|
|
ullint
|
|
|
|
trx_i_s_cache_get_rows_used(
|
|
|
|
/*========================*/
|
|
|
|
/* out: number of rows */
|
|
|
|
trx_i_s_cache_t* cache, /* in: cache */
|
|
|
|
enum i_s_table table) /* in: which table */
|
|
|
|
{
|
|
|
|
i_s_table_cache_t* table_cache;
|
|
|
|
|
|
|
|
table_cache = cache_select_table(cache, table);
|
|
|
|
|
|
|
|
return(table_cache->rows_used);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Retrieves the nth row (zero-based) in the cache for a given
|
|
|
|
INFORMATION SCHEMA table. */
|
|
|
|
|
|
|
|
void*
|
|
|
|
trx_i_s_cache_get_nth_row(
|
|
|
|
/*======================*/
|
|
|
|
/* out: row */
|
|
|
|
trx_i_s_cache_t* cache, /* in: cache */
|
|
|
|
enum i_s_table table, /* in: which table */
|
|
|
|
ulint n) /* in: row number */
|
|
|
|
{
|
|
|
|
i_s_table_cache_t* table_cache;
|
|
|
|
ulint i;
|
|
|
|
void* row;
|
|
|
|
|
|
|
|
table_cache = cache_select_table(cache, table);
|
|
|
|
|
|
|
|
ut_a(n < table_cache->rows_used);
|
|
|
|
|
|
|
|
row = NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < MEM_CHUNKS_IN_TABLE_CACHE; i++) {
|
|
|
|
|
|
|
|
if (table_cache->chunks[i].offset
|
|
|
|
+ table_cache->chunks[i].rows_allocd > n) {
|
|
|
|
|
|
|
|
row = (char*) table_cache->chunks[i].base
|
|
|
|
+ (n - table_cache->chunks[i].offset)
|
|
|
|
* table_cache->row_size;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ut_a(row != NULL);
|
|
|
|
|
|
|
|
return(row);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
Crafts a lock id string from a i_s_locks_row_t object. Returns its
|
|
|
|
second argument. This function aborts if there is not enough space in
|
|
|
|
lock_id. Be sure to provide at least TRX_I_S_LOCK_ID_MAX_LEN if you want
|
|
|
|
to be 100% sure that it will not abort. */
|
|
|
|
|
|
|
|
char*
|
|
|
|
trx_i_s_create_lock_id(
|
|
|
|
/*===================*/
|
|
|
|
/* out: resulting lock id */
|
|
|
|
const i_s_locks_row_t* row, /* in: innodb_locks row */
|
|
|
|
char* lock_id,/* out: resulting lock_id */
|
|
|
|
ulint lock_id_size)/* in: size of the lock id
|
|
|
|
buffer */
|
|
|
|
{
|
|
|
|
int res_len;
|
|
|
|
|
|
|
|
/* please adjust TRX_I_S_LOCK_ID_MAX_LEN if you change this */
|
|
|
|
|
|
|
|
if (row->lock_space != ULINT_UNDEFINED) {
|
|
|
|
/* record lock */
|
|
|
|
res_len = ut_snprintf(lock_id, lock_id_size,
|
|
|
|
"%llu:%lu:%lu:%lu",
|
|
|
|
row->lock_trx_id, row->lock_space,
|
|
|
|
row->lock_page, row->lock_rec);
|
|
|
|
} else {
|
|
|
|
/* table lock */
|
|
|
|
res_len = ut_snprintf(lock_id, lock_id_size,
|
|
|
|
"%llu:%llu",
|
|
|
|
row->lock_trx_id,
|
|
|
|
row->lock_table_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* the typecast is safe because snprintf(3) never returns
|
|
|
|
negative result */
|
|
|
|
ut_a(res_len >= 0);
|
|
|
|
ut_a((ulint) res_len < lock_id_size);
|
|
|
|
|
|
|
|
return(lock_id);
|
|
|
|
}
|