1
/*****************************************************************************
3
Copyright (C) 2007, 2009, Innobase Oy. All Rights Reserved.
5
This program is free software; you can redistribute it and/or modify it under
6
the terms of the GNU General Public License as published by the Free Software
7
Foundation; version 2 of the License.
9
This program is distributed in the hope that it will be useful, but WITHOUT
10
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
11
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
13
You should have received a copy of the GNU General Public License along with
14
this program; if not, write to the Free Software Foundation, Inc., 51 Franklin
15
St, Fifth Floor, Boston, MA 02110-1301 USA
17
*****************************************************************************/
21
#include "data_dictionary.h"
23
#include "drizzled/current_session.h"
26
#include "trx0trx.h" /* for TRX_QUE_STATE_STR_MAX_LEN */
27
#include "buf0buddy.h" /* for i_s_cmpmem */
28
#include "buf0buf.h" /* for buf_pool and PAGE_ZIP_MIN_SIZE */
29
#include "ha_prototypes.h" /* for innobase_convert_name() */
30
#include "srv0start.h" /* for srv_was_started */
31
#include "btr0pcur.h" /* for file sys_tables related info. */
32
#include "btr0types.h"
33
#include "dict0load.h" /* for file sys_tables related info. */
35
#include "dict0types.h"
36
#include "handler0vars.h"
38
using namespace drizzled;
40
InnodbSysTablesTool::InnodbSysTablesTool() :
41
plugin::TableFunction("DATA_DICTIONARY", "INNODB_SYS_TABLES")
43
add_field("TABLE_ID", plugin::TableFunction::NUMBER, 0, false);
44
add_field("NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
45
add_field("FLAG", plugin::TableFunction::NUMBER, 0, false);
46
add_field("N_COLS", plugin::TableFunction::NUMBER, 0, false);
47
add_field("SPACE", plugin::TableFunction::NUMBER, 0, false);
50
InnodbSysTablesTool::Generator::Generator(Field **arg) :
51
plugin::TableFunction::Generator(arg)
56
bool InnodbSysTablesTool::Generator::populate()
60
heap = mem_heap_create(1000);
61
mutex_enter(&(dict_sys->mutex));
64
rec = dict_startscan_system(&pcur, &mtr, SYS_TABLES);
68
/* Get the next record */
69
mutex_enter(&dict_sys->mutex);
71
rec = dict_getnext_system(&pcur, &mtr);
77
mutex_exit(&dict_sys->mutex);
83
dict_table_t* table_rec;
85
/* Create and populate a dict_table_t structure with
86
information from SYS_TABLES row */
87
err_msg = dict_process_sys_tables_rec(heap, rec, &table_rec,
88
DICT_TABLE_LOAD_FROM_RECORD);
91
mutex_exit(&dict_sys->mutex);
95
push(table_rec->name);
96
push(static_cast<uint64_t>(table_rec->flags));
97
push(static_cast<uint64_t>(table_rec->n_cols));
98
push(static_cast<uint64_t>(table_rec->space));
100
/* push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_WARN,
101
ER_CANT_FIND_SYSTEM_REC,
105
/* Since dict_process_sys_tables_rec() is called with
106
DICT_TABLE_LOAD_FROM_RECORD, the table_rec is created in
107
dict_process_sys_tables_rec(), we will need to free it */
109
dict_mem_table_free(table_rec);
112
mem_heap_empty(heap);
117
InnodbSysTableStatsTool::InnodbSysTableStatsTool() :
118
plugin::TableFunction("DATA_DICTIONARY", "INNODB_SYS_TABLESTATS")
120
add_field("TABLE_ID", plugin::TableFunction::NUMBER, 0, false);
121
add_field("NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
122
add_field("STATS_INITIALIZED", plugin::TableFunction::STRING, NAME_LEN + 1, false);
123
add_field("NUM_ROWS", plugin::TableFunction::NUMBER, 0, false);
124
add_field("CLUST_INDEX_SIZE", plugin::TableFunction::NUMBER, 0, false);
125
add_field("OTHER_INDEX_SIZE", plugin::TableFunction::NUMBER, 0, false);
126
add_field("MODIFIED_COUNTER", plugin::TableFunction::NUMBER, 0, false);
127
add_field("AUTOINC", plugin::TableFunction::NUMBER, 0, false);
128
add_field("HANDLES_OPENED", plugin::TableFunction::NUMBER, 0, false);
131
InnodbSysTableStatsTool::Generator::Generator(Field **arg) :
132
plugin::TableFunction::Generator(arg)
137
bool InnodbSysTableStatsTool::Generator::populate()
141
heap = mem_heap_create(1000);
142
mutex_enter(&dict_sys->mutex);
144
rec = dict_startscan_system(&pcur, &mtr, SYS_TABLES);
148
/* Get the next record */
149
mutex_enter(&dict_sys->mutex);
151
rec = dict_getnext_system(&pcur, &mtr);
157
mutex_exit(&dict_sys->mutex);
163
dict_table_t* table_rec;
165
/* Fetch the dict_table_t structure corresponding to
166
this SYS_TABLES record */
167
err_msg = dict_process_sys_tables_rec(heap, rec, &table_rec,
168
DICT_TABLE_LOAD_FROM_CACHE);
171
mutex_exit(&dict_sys->mutex);
175
push(table_rec->name);
176
if (table_rec->stat_initialized)
179
push("Uninitialized");
180
push(table_rec->stat_n_rows);
181
push(static_cast<uint64_t>(table_rec->stat_clustered_index_size));
182
push(static_cast<uint64_t>(table_rec->stat_sum_of_other_index_sizes));
183
push(static_cast<uint64_t>(table_rec->stat_modified_counter));
184
push(table_rec->autoinc);
185
push(static_cast<uint64_t>(table_rec->n_mysql_handles_opened));
187
/* push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
188
ER_CANT_FIND_SYSTEM_REC,
192
mem_heap_empty(heap);
197
InnodbSysIndexesTool::InnodbSysIndexesTool() :
198
plugin::TableFunction("DATA_DICTIONARY", "INNODB_SYS_INDEXES")
200
add_field("INDEX_ID", plugin::TableFunction::NUMBER, 0, false);
201
add_field("NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
202
add_field("TABLE_ID", plugin::TableFunction::NUMBER, 0, false);
203
add_field("TYPE", plugin::TableFunction::NUMBER, 0, false);
204
add_field("N_FIELDS", plugin::TableFunction::NUMBER, 0, false);
205
add_field("PAGE_NO", plugin::TableFunction::NUMBER, 0, false);
206
add_field("SPACE", plugin::TableFunction::NUMBER, 0, false);
209
InnodbSysIndexesTool::Generator::Generator(Field **arg) :
210
plugin::TableFunction::Generator(arg)
215
bool InnodbSysIndexesTool::Generator::populate()
219
heap = mem_heap_create(1000);
220
mutex_enter(&dict_sys->mutex);
223
/* Start scan the SYS_INDEXES table */
224
rec = dict_startscan_system(&pcur, &mtr, SYS_INDEXES);
228
/* Get the next record */
229
mutex_enter(&dict_sys->mutex);
231
rec = dict_getnext_system(&pcur, &mtr);
237
mutex_exit(&dict_sys->mutex);
242
const char* err_msg;;
244
dict_index_t index_rec;
246
/* Populate a dict_index_t structure with information from
248
err_msg = dict_process_sys_indexes_rec(heap, rec, &index_rec,
252
mutex_exit(&dict_sys->mutex);
255
push(index_rec.name);
256
push(static_cast<uint64_t>(table_id));
257
push(static_cast<uint64_t>(index_rec.type));
258
push(static_cast<uint64_t>(index_rec.n_fields));
259
push(static_cast<uint64_t>(index_rec.page));
260
push(static_cast<uint64_t>(index_rec.space));
262
/* push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
263
ER_CANT_FIND_SYSTEM_REC,
267
mem_heap_empty(heap);
272
mutex_exit(&dict_sys->mutex);
280
InnodbSysColumnsTool::InnodbSysColumnsTool() :
281
plugin::TableFunction("DATA_DICTIONARY", "INNODB_SYS_COLUMNS")
283
add_field("TABLE_ID", plugin::TableFunction::NUMBER, 0, false);
284
add_field("NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
285
add_field("POS", plugin::TableFunction::NUMBER, 0, false);
286
add_field("MTYPE", plugin::TableFunction::NUMBER, 0, false);
287
add_field("PRTYPE", plugin::TableFunction::NUMBER, 0, false);
288
add_field("LEN", plugin::TableFunction::NUMBER, 0, false);
291
InnodbSysColumnsTool::Generator::Generator(Field **arg) :
292
plugin::TableFunction::Generator(arg)
297
bool InnodbSysColumnsTool::Generator::populate()
301
heap = mem_heap_create(1000);
302
mutex_enter(&dict_sys->mutex);
304
rec = dict_startscan_system(&pcur, &mtr, SYS_COLUMNS);
308
/* Get the next record */
309
mutex_enter(&dict_sys->mutex);
311
rec = dict_getnext_system(&pcur, &mtr);
317
mutex_exit(&dict_sys->mutex);
323
dict_col_t column_rec;
325
const char* col_name;
327
/* populate a dict_col_t structure with information from
329
err_msg = dict_process_sys_columns_rec(heap, rec, &column_rec,
330
&table_id, &col_name);
333
mutex_exit(&dict_sys->mutex);
338
push(static_cast<uint64_t>(column_rec.ind));
339
push(static_cast<uint64_t>(column_rec.mtype));
340
push(static_cast<uint64_t>(column_rec.prtype));
341
push(static_cast<uint64_t>(column_rec.len));
343
/* push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
344
ER_CANT_FIND_SYSTEM_REC,
348
mem_heap_empty(heap);
353
InnodbSysFieldsTool::InnodbSysFieldsTool() :
354
plugin::TableFunction("DATA_DICTIONARY", "INNODB_SYS_FIELDS")
356
add_field("INDEX_ID", plugin::TableFunction::NUMBER, 0, false);
357
add_field("NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
358
add_field("POS", plugin::TableFunction::NUMBER, 0, false);
361
InnodbSysFieldsTool::Generator::Generator(Field **arg) :
362
plugin::TableFunction::Generator(arg)
367
bool InnodbSysFieldsTool::Generator::populate()
371
heap = mem_heap_create(1000);
372
mutex_enter(&dict_sys->mutex);
375
/* will save last index id so that we know whether we move to
376
the next index. This is used to calculate prefix length */
379
rec = dict_startscan_system(&pcur, &mtr, SYS_FIELDS);
383
/* Get the next record */
384
mutex_enter(&dict_sys->mutex);
386
rec = dict_getnext_system(&pcur, &mtr);
392
mutex_exit(&dict_sys->mutex);
401
dict_field_t field_rec;
403
/* Populate a dict_field_t structure with information from
405
err_msg = dict_process_sys_fields_rec(heap, rec, &field_rec,
406
&pos, &index_id, last_id);
409
mutex_exit(&dict_sys->mutex);
413
push(field_rec.name);
414
push(static_cast<uint64_t>(pos));
418
/* push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
419
ER_CANT_FIND_SYSTEM_REC,
423
mem_heap_empty(heap);
428
InnodbSysForeignTool::InnodbSysForeignTool() :
429
plugin::TableFunction("DATA_DICTIONARY", "INNODB_SYS_FOREIGN")
431
add_field("ID", plugin::TableFunction::STRING, NAME_LEN + 1, false);
432
add_field("FOR_NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
433
add_field("REF_NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
434
add_field("N_COLS", plugin::TableFunction::NUMBER, 0, false);
435
add_field("TYPE", plugin::TableFunction::NUMBER, 0, false);
438
InnodbSysForeignTool::Generator::Generator(Field **arg) :
439
plugin::TableFunction::Generator(arg)
444
bool InnodbSysForeignTool::Generator::populate()
448
heap = mem_heap_create(1000);
449
mutex_enter(&dict_sys->mutex);
452
rec = dict_startscan_system(&pcur, &mtr, SYS_FOREIGN);
456
/* Get the next record */
458
mutex_enter(&dict_sys->mutex);
459
rec = dict_getnext_system(&pcur, &mtr);
465
mutex_exit(&dict_sys->mutex);
471
dict_foreign_t foreign_rec;
473
/* Populate a dict_foreign_t structure with information from
475
err_msg = dict_process_sys_foreign_rec(heap, rec, &foreign_rec);
478
mutex_exit(&dict_sys->mutex);
481
push(foreign_rec.id);
482
push(foreign_rec.foreign_table_name);
483
push(foreign_rec.referenced_table_name);
484
push(static_cast<uint64_t>(foreign_rec.n_fields));
485
push(static_cast<uint64_t>(foreign_rec.type));
487
/* push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
488
ER_CANT_FIND_SYSTEM_REC,
492
mem_heap_empty(heap);
497
InnodbSysForeignColsTool::InnodbSysForeignColsTool() :
498
plugin::TableFunction("DATA_DICTIONARY", "INNODB_SYS_FOREIGN_COLS")
500
add_field("ID", plugin::TableFunction::STRING, NAME_LEN + 1, false);
501
add_field("FOR_COL_NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
502
add_field("REF_COL_NAME", plugin::TableFunction::STRING, NAME_LEN + 1, false);
503
add_field("POS", plugin::TableFunction::NUMBER, 0, false);
506
InnodbSysForeignColsTool::Generator::Generator(Field **arg) :
507
plugin::TableFunction::Generator(arg)
512
bool InnodbSysForeignColsTool::Generator::populate()
516
heap = mem_heap_create(1000);
517
mutex_enter(&dict_sys->mutex);
520
rec = dict_startscan_system(&pcur, &mtr, SYS_FOREIGN_COLS);
524
/* Get the next record */
525
mutex_enter(&dict_sys->mutex);
527
rec = dict_getnext_system(&pcur, &mtr);
533
mutex_exit(&dict_sys->mutex);
541
const char* for_col_name;
542
const char* ref_col_name;
545
/* Extract necessary information from a SYS_FOREIGN_COLS row */
546
err_msg = dict_process_sys_foreign_col_rec(heap, rec, &name, &for_col_name,
547
&ref_col_name, &pos);
550
mutex_exit(&dict_sys->mutex);
556
push(static_cast<uint64_t>(pos));
558
/* push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
559
ER_CANT_FIND_SYSTEM_REC,
564
mem_heap_empty(heap);
570
* Fill the dynamic table data_dictionary.INNODB_CMP and INNODB_CMP_RESET
573
CmpTool::CmpTool(bool in_reset) :
574
plugin::TableFunction("DATA_DICTIONARY", in_reset ? "INNODB_CMP_RESET" : "INNODB_CMP"),
575
outer_reset(in_reset)
577
add_field("PAGE_SIZE", plugin::TableFunction::NUMBER, 0, false);
578
add_field("COMPRESS_OPS", plugin::TableFunction::NUMBER, 0, false);
579
add_field("COMPRESS_OPS_OK", plugin::TableFunction::NUMBER, 0, false);
580
add_field("COMPRESS_TIME", plugin::TableFunction::NUMBER, 0, false);
581
add_field("UNCOMPRESS_OPS", plugin::TableFunction::NUMBER, 0, false);
582
add_field("UNCOMPRESS_TIME", plugin::TableFunction::NUMBER, 0, false);
585
CmpTool::Generator::Generator(Field **arg, bool in_reset) :
586
plugin::TableFunction::Generator(arg),
588
inner_reset(in_reset)
592
bool CmpTool::Generator::populate()
594
if (record_number == (PAGE_ZIP_NUM_SSIZE - 1))
599
page_zip_stat_t* zip_stat = &page_zip_stat[record_number];
601
push(static_cast<uint64_t>(PAGE_ZIP_MIN_SIZE << record_number));
603
/* The cumulated counts are not protected by any
604
mutex. Thus, some operation in page0zip.c could
605
increment a counter between the time we read it and
606
clear it. We could introduce mutex protection, but it
607
could cause a measureable performance hit in
609
push(static_cast<uint64_t>(zip_stat->compressed));
610
push(static_cast<uint64_t>(zip_stat->compressed_ok));
611
push(zip_stat->compressed_usec / 1000000);
612
push(static_cast<uint64_t>(zip_stat->decompressed));
613
push(zip_stat->decompressed_usec / 1000000);
617
memset(zip_stat, 0, sizeof *zip_stat);
626
* Fill the dynamic table data_dictionary.INNODB_CMPMEM and INNODB_CMPMEM_RESET
629
CmpmemTool::CmpmemTool(bool in_reset) :
630
plugin::TableFunction("DATA_DICTIONARY", in_reset ? "INNODB_CMPMEM_RESET" : "INNODB_CMPMEM"),
631
outer_reset(in_reset)
633
add_field("BUF_POOL", plugin::TableFunction::NUMBER, 0, false);
634
add_field("PAGE_SIZE", plugin::TableFunction::NUMBER, 0, false);
635
add_field("PAGES_USED", plugin::TableFunction::NUMBER, 0, false);
636
add_field("PAGES_FREE", plugin::TableFunction::NUMBER, 0, false);
637
add_field("RELOCATION_OPS", plugin::TableFunction::NUMBER, 0, false);
638
add_field("RELOCATION_TIME", plugin::TableFunction::NUMBER, 0, false);
641
CmpmemTool::Generator::Generator(Field **arg, bool in_reset) :
642
plugin::TableFunction::Generator(arg),
644
inner_reset(in_reset)
648
CmpmemTool::Generator::~Generator()
652
bool CmpmemTool::Generator::populate()
654
if (record_number >= (BUF_BUDDY_SIZES+1)*srv_buf_pool_instances)
659
uint32_t buddy_nr= record_number % (BUF_BUDDY_SIZES+1);
660
uint32_t buf_pool_nr= (record_number/(BUF_BUDDY_SIZES+1));
662
buf_pool_t *buf_pool= buf_pool_from_array(buf_pool_nr);
664
buf_pool_mutex_enter(buf_pool);
666
buf_buddy_stat_t* buddy_stat = &buf_pool->buddy_stat[buddy_nr];
669
push(static_cast<uint64_t>(buf_pool_nr));
670
push(static_cast<uint64_t>(BUF_BUDDY_LOW << buddy_nr));
671
push(static_cast<uint64_t>(buddy_stat->used));
674
uint64_t pages_free= (UNIV_LIKELY(buddy_nr < BUF_BUDDY_SIZES) ? UT_LIST_GET_LEN(buf_pool->zip_free[buddy_nr]) : 0);
677
push(buddy_stat->relocated);
678
push(buddy_stat->relocated_usec / 1000000);
683
buddy_stat->relocated = 0;
684
buddy_stat->relocated_usec = 0;
687
buf_pool_mutex_exit(buf_pool);
694
* Fill the dynamic table data_dictionary.INNODB_TRX INNODB_LOCKS INNODB_LOCK_WAITS
697
InnodbTrxTool::InnodbTrxTool(const char* in_table_name) :
698
plugin::TableFunction("DATA_DICTIONARY", in_table_name),
699
table_name(in_table_name)
701
if (innobase_strcasecmp(table_name, "INNODB_TRX") == 0)
704
add_field("TRX_STATE");
705
add_field("TRX_STARTED", plugin::TableFunction::NUMBER, 0, false);
706
add_field("TRX_REQUESTED_LOCK_ID");
707
add_field("TRX_WAIT_STARTED", plugin::TableFunction::NUMBER, 0, false);
708
add_field("TRX_WEIGHT", plugin::TableFunction::NUMBER, 0, false);
709
add_field("TRX_DRIZZLE_THREAD_ID", plugin::TableFunction::NUMBER, 0, false);
710
add_field("TRX_QUERY", plugin::TableFunction::STRING, TRX_I_S_TRX_QUERY_MAX_LEN, true);
711
add_field("TRX_OPERATION_STATE", plugin::TableFunction::STRING, TRX_I_S_TRX_OP_STATE_MAX_LEN, true);
712
// add_field("TRX_TABLES_IN_USE", plugin::TableFunction::NUMBER, 0, false);
713
add_field("TRX_TABLES_LOCKED", plugin::TableFunction::NUMBER, 0, false);
714
add_field("TRX_LOCK_STRUCTS", plugin::TableFunction::NUMBER, 0, false);
715
add_field("TRX_LOCK_MEMORY_BYTES", plugin::TableFunction::NUMBER, 0, false);
716
add_field("TRX_ROWS_LOCKED", plugin::TableFunction::NUMBER, 0, false);
717
add_field("TRX_ROWS_MODIFIED", plugin::TableFunction::NUMBER, 0, false);
718
add_field("TRX_CONCURRENCY_TICKETS", plugin::TableFunction::NUMBER, 0, false);
719
add_field("TRX_ISOLATION_LEVEL", plugin::TableFunction::STRING, TRX_I_S_TRX_ISOLATION_LEVEL_MAX_LEN, false);
720
add_field("TRX_UNIQUE_CHECKS", plugin::TableFunction::NUMBER, 0, false);
721
add_field("TRX_FOREIGN_KEY_CHECKS", plugin::TableFunction::NUMBER, 0, false);
722
add_field("TRX_LAST_FOREIGN_KEY_ERROR", plugin::TableFunction::STRING,
723
TRX_I_S_TRX_FK_ERROR_MAX_LEN, true);
724
add_field("TRX_ADAPTIVE_HASH_LATCHED", plugin::TableFunction::NUMBER, 0, false);
725
add_field("TRX_ADAPTIVE_HASH_TIMEOUT", plugin::TableFunction::NUMBER, 0, false);
727
else if (innobase_strcasecmp(table_name, "INNODB_LOCKS") == 0)
729
add_field("LOCK_ID");
730
add_field("LOCK_TRX_ID");
731
add_field("LOCK_MODE");
732
add_field("LOCK_TYPE");
733
add_field("LOCK_TABLE");
734
add_field("LOCK_INDEX");
735
add_field("LOCK_SPACE", plugin::TableFunction::NUMBER, 0, false);
736
add_field("LOCK_PAGE", plugin::TableFunction::NUMBER, 0, false);
737
add_field("LOCK_REC", plugin::TableFunction::NUMBER, 0, false);
738
add_field("LOCK_DATA");
740
else if (innobase_strcasecmp(table_name, "INNODB_LOCK_WAITS") == 0)
742
add_field("REQUESTING_TRX_ID");
743
add_field("REQUESTED_LOCK_ID");
744
add_field("BLOCKING_TRX_ID");
745
add_field("BLOCKING_LOCK_ID");
749
InnodbTrxTool::Generator::Generator(Field **arg, const char* in_table_name) :
750
plugin::TableFunction::Generator(arg),
751
table_name(in_table_name)
753
/* update the cache */
754
trx_i_s_cache_start_write(trx_i_s_cache);
755
trx_i_s_possibly_fetch_data_into_cache(trx_i_s_cache);
756
trx_i_s_cache_end_write(trx_i_s_cache);
758
if (trx_i_s_cache_is_truncated(trx_i_s_cache))
760
errmsg_printf(error::ERROR, _("Warning: data in %s truncated due to memory limit of %d bytes\n"),
761
table_name, TRX_I_S_MEM_LIMIT);
764
trx_i_s_cache_start_read(trx_i_s_cache);
766
if (innobase_strcasecmp(table_name, "INNODB_TRX") == 0)
767
number_rows= trx_i_s_cache_get_rows_used(trx_i_s_cache, I_S_INNODB_TRX);
768
else if (innobase_strcasecmp(table_name, "INNODB_LOCKS") == 0)
769
number_rows= trx_i_s_cache_get_rows_used(trx_i_s_cache, I_S_INNODB_LOCKS);
770
else if (innobase_strcasecmp(table_name, "INNODB_LOCK_WAITS") == 0)
771
number_rows= trx_i_s_cache_get_rows_used(trx_i_s_cache, I_S_INNODB_LOCK_WAITS);
776
InnodbTrxTool::Generator::~Generator()
778
trx_i_s_cache_end_read(trx_i_s_cache);
781
bool InnodbTrxTool::Generator::populate()
783
if (record_number == number_rows)
788
if (innobase_strcasecmp(table_name, "INNODB_TRX") == 0)
790
populate_innodb_trx();
792
else if (innobase_strcasecmp(table_name, "INNODB_LOCKS") == 0)
794
populate_innodb_locks();
796
else if (innobase_strcasecmp(table_name, "INNODB_LOCK_WAITS") == 0)
798
populate_innodb_lock_waits();
809
void InnodbTrxTool::Generator::populate_innodb_locks()
812
char lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
813
i_s_locks_row_t* row;
815
/* note that the decoded database or table name is
816
never expected to be longer than NAME_LEN;
817
NAME_LEN for database name
818
2 for surrounding quotes around database name
819
NAME_LEN for table name
820
2 for surrounding quotes around table name
821
1 for the separating dot (.)
822
9 for the #mysql50# prefix
825
char buf[2 * NAME_LEN + 14];
828
char lock_trx_id[TRX_ID_MAX_LEN + 1];
830
row = (i_s_locks_row_t*)
831
trx_i_s_cache_get_nth_row(
832
trx_i_s_cache, I_S_INNODB_LOCKS, record_number);
835
trx_i_s_create_lock_id(row, lock_id, sizeof(lock_id));
838
ut_snprintf(lock_trx_id, sizeof(lock_trx_id),
839
TRX_ID_FMT, row->lock_trx_id);
842
push(row->lock_mode);
843
push(row->lock_type);
845
bufend = innobase_convert_name(buf, sizeof(buf),
847
strlen(row->lock_table),
848
&getSession(), TRUE);
851
if (row->lock_index != NULL)
853
bufend = innobase_convert_name(buf, sizeof(buf),
855
strlen(row->lock_index),
856
&getSession(), FALSE);
864
push(static_cast<uint64_t>(row->lock_space));
865
push(static_cast<uint64_t>(row->lock_page));
866
push(static_cast<uint64_t>(row->lock_rec));
867
push(row->lock_data);
870
void InnodbTrxTool::Generator::populate_innodb_trx()
872
char lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
874
char trx_id[TRX_ID_MAX_LEN + 1];
875
row = (i_s_trx_row_t*) trx_i_s_cache_get_nth_row(trx_i_s_cache, I_S_INNODB_TRX, record_number);
878
ut_snprintf(trx_id, sizeof(trx_id), TRX_ID_FMT, row->trx_id);
881
push(row->trx_state);
882
push(static_cast<uint64_t>(row->trx_started));
884
if (row->trx_wait_started != 0)
886
push(trx_i_s_create_lock_id(row->requested_lock_row, lock_id, sizeof(lock_id)));
887
push(static_cast<uint64_t>(row->trx_wait_started));
891
push(static_cast<uint64_t>(0));
892
push(static_cast<uint64_t>(0));
895
push(static_cast<int64_t>(row->trx_weight));
896
push(static_cast<uint64_t>(row->trx_mysql_thread_id));
898
push(row->trx_query);
902
if (row->trx_operation_state)
903
push(row->trx_operation_state);
907
// push(row->trx_tables_in_use);
908
push(static_cast<uint64_t>(row->trx_tables_locked));
909
push(static_cast<uint64_t>(row->trx_lock_structs));
910
push(static_cast<uint64_t>(row->trx_lock_memory_bytes));
911
push(static_cast<uint64_t>(row->trx_rows_locked));
912
push(static_cast<uint64_t>(row->trx_rows_modified));
913
push(static_cast<uint64_t>(row->trx_concurrency_tickets));
914
push(row->trx_isolation_level);
915
push(static_cast<uint64_t>(row->trx_unique_checks));
916
push(static_cast<uint64_t>(row->trx_foreign_key_checks));
917
if (row->trx_foreign_key_error)
918
push(row->trx_foreign_key_error);
922
push(static_cast<uint64_t>(row->trx_has_search_latch));
923
push(static_cast<uint64_t>(row->trx_search_latch_timeout));
926
void InnodbTrxTool::Generator::populate_innodb_lock_waits()
928
char requested_lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
929
char blocking_lock_id[TRX_I_S_LOCK_ID_MAX_LEN + 1];
931
i_s_lock_waits_row_t* row;
933
char requesting_trx_id[TRX_ID_MAX_LEN + 1];
934
char blocking_trx_id[TRX_ID_MAX_LEN + 1];
936
row = (i_s_lock_waits_row_t*)
937
trx_i_s_cache_get_nth_row(
938
trx_i_s_cache, I_S_INNODB_LOCK_WAITS, record_number);
940
ut_snprintf(requesting_trx_id, sizeof(requesting_trx_id),
941
TRX_ID_FMT, row->requested_lock_row->lock_trx_id);
942
push(requesting_trx_id);
944
push(trx_i_s_create_lock_id(row->requested_lock_row, requested_lock_id,
945
sizeof(requested_lock_id)));
947
ut_snprintf(blocking_trx_id, sizeof(blocking_trx_id),
948
TRX_ID_FMT, row->blocking_lock_row->lock_trx_id);
949
push(blocking_trx_id);
951
push(trx_i_s_create_lock_id(row->blocking_lock_row, blocking_lock_id,
952
sizeof(blocking_lock_id)));